要点

  • 用A、B、C一般式确定每条直线
  • 将合法的圆心中点存到每条直线所属的vector中
  • 枚举所有线段,二分后\(O(1)\)得到其中存在多少答案,累加
#include <cstdio>
#include <cmath>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <iomanip>
#include <vector>
#include <unordered_map>
using namespace std; typedef long long ll;
const int maxn = 3e5 + 5, maxm = 1550;
const ll inf = 1e18; struct Point {
ll x, y;
Point() {}
Point(ll a, ll b):x(a), y(b) {}
};
struct Circle {
Point p;
ll r;
Circle() {}
Circle(Point a, ll b):p(a), r(b) {}
}; int n, m, hashcnt;
Point a[maxn], b[maxn];//segments
Circle c[maxm];//circles
unordered_map<ll, int> mp;//<{A, B, C}, id>
vector<ll> v[maxn];//v[id]
ll A, B, C;//a few times used
ll ans;
int ID[maxn]; ll sqr(ll x) {
return x * x;
} ll dis(int i, int j) {//distance ^ 2
return sqr(c[i].p.x - c[j].p.x) + sqr(c[i].p.y - c[j].p.y);
} ll gcd(ll a, ll b) {//exist zero: return
if (!a || !b) return a + b;
return gcd(b, a % b);
} ll Cross(Point A, Point B) {
return A.x * B.y - A.y * B.x;
} void Deal(ll &A, ll &B, ll &C) {//unique the line
ll q = gcd(gcd(abs(A), abs(B)), abs(C));
A /= q, B /= q, C /= q;
if (A == 0 && B < 0) B = -B, C = -C;
else if (A < 0) A = -A, B = -B, C = -C;
} ll hashi(ll A, ll B, ll C) {//map is TLE, so I hash it
return A * (ll)(1e12) + B * (ll)(1e6) + C;
} void Hash(int i, ll A, ll B, ll C) {
ll d = hashi(A, B, C);
if (!mp.count(d)) {
mp[d] = ++hashcnt;
}
int id = mp[d];
ID[i] = id;
} void calc(Point a, Point b) {// get A, B, C
A = b.y - a.y, B = a.x - b.x;
C = Cross(b, a);//Ax + By + C = 0
Deal(A, B, C);
} int main() {
scanf("%d %d", &n, &m);
for (int i = 1; i <= n; i++) {
scanf("%lld %lld %lld %lld", &a[i].x, &a[i].y, &b[i].x, &b[i].y);
a[i].x *= 2, a[i].y *= 2, b[i].x *= 2, b[i].y *= 2;//for line 99
if (a[i].x > b[i].x) swap(a[i], b[i]);//for line 109 & 110
calc(a[i], b[i]);
Hash(i, A, B, C);
}
for (int i = 1; i <= m; i++) {
scanf("%lld %lld %lld", &c[i].p.x, &c[i].p.y, &c[i].r);
c[i].p.x *= 2, c[i].p.y *= 2, c[i].r *= 2;
for (int j = 1; j < i; j++)
if (c[i].r == c[j].r && dis(i, j) > 4LL * sqr(c[i].r)) {//if eyes
calc(c[i].p, c[j].p);
ll A1 = B * 2, B1 = -A * 2;
ll C1 = A * (c[i].p.y + c[j].p.y) - B * (c[i].p.x + c[j].p.x);
Deal(A1, B1, C1);
ll d = hashi(A1, B1, C1);
if (!mp.count(d)) continue;
ll x = (c[i].p.x + c[j].p.x) / 2;//line 99: should avoid double
v[mp[d]].emplace_back(x);
}
}
for (int i = 1; i <= hashcnt; i++) {
v[i].emplace_back(inf);
sort(v[i].begin(), v[i].end());
}
for (int i = 1; i <= n; i++) {
int id = ID[i];
int l = lower_bound(v[id].begin(), v[id].end(), a[i].x) - v[id].begin();//line 109
int r = upper_bound(v[id].begin(), v[id].end(), b[i].x) - v[id].begin();//line 110
ans += r - l;
}
return !printf("%lld\n", ans);
}

Codeforces 350D(计算几何)的更多相关文章

  1. Codeforces Gym100543B 计算几何 凸包 线段树 二分/三分 卡常

    原文链接https://www.cnblogs.com/zhouzhendong/p/CF-Gym100543B.html 题目传送门 - CF-Gym100543B 题意 给定一个折线图,对于每一条 ...

  2. Codeforces Round #335 (Div. 1) C. Freelancer's Dreams 计算几何

    C. Freelancer's Dreams Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.codeforces.com/contes ...

  3. Codeforces 749B:Parallelogram is Back(计算几何)

    http://codeforces.com/problemset/problem/749/B 题意:已知平行四边形三个顶点,求另外一个顶点可能的位置. 思路:用向量来做. #include <c ...

  4. Codeforces Round #339 (Div. 1) A. Peter and Snow Blower 计算几何

    A. Peter and Snow Blower 题目连接: http://www.codeforces.com/contest/613/problem/A Description Peter got ...

  5. Codeforces Round #524 (Div. 2) C. Masha and two friends(思维+计算几何?)

    传送门 https://www.cnblogs.com/violet-acmer/p/10146350.html 题意: 有一块 n*m 的棋盘,初始,黑白块相间排列,且左下角为白块. 给出两个区间[ ...

  6. Codeforces 528E Triangles 3000 - 计算几何

    题目传送门 传送点I 传送点II 传送点III 题目大意 给定$n$的平面上的直线,保证没有三条直线共点,两条直线平行.问随机选出3条直线交成的三角形面积的期望. 显然$S=\frac{1}{2}ah ...

  7. Codeforces Round #284 (Div. 1) A. Crazy Town 计算几何

    A. Crazy Town 题目连接: http://codeforces.com/contest/498/problem/A Description Crazy Town is a plane on ...

  8. Codeforces Round #357 (Div. 2) E. Runaway to a Shadow 计算几何

    E. Runaway to a Shadow 题目连接: http://www.codeforces.com/contest/681/problem/E Description Dima is liv ...

  9. Codeforces Beta Round #1 C. Ancient Berland Circus 计算几何

    C. Ancient Berland Circus 题目连接: http://www.codeforces.com/contest/1/problem/C Description Nowadays a ...

随机推荐

  1. 搭建DNS服务器-bind

    1. 安装 yum install -y bind-chroot yum install -y bind-utils service named-chroot start    2. 修改配置 增加一 ...

  2. BZOJ1345:[Baltic2007]序列问题

    浅谈栈:https://www.cnblogs.com/AKMer/p/10278222.html 题目传送门:https://lydsy.com/JudgeOnline/problem.php?id ...

  3. CF 293E Close Vertices——点分治

    题目:http://codeforces.com/contest/293/problem/E 仍旧是点分治.用容斥,w的限制用排序+两个指针解决, l 的限制就用树状数组.有0的话就都+1,相对大小不 ...

  4. Navicat生成数据库结构同步SQL

    作为一个苦逼的技术男,在做开发的时候经常会遇见程序版本升级,数据库结构变化.我们需要一个快捷的方式让客户尽快从旧版本数据库结构更新至新版本数据库结构.如果每做一次改动我们就记录一下当然是好事,但是万一 ...

  5. jexus入门

    参考:https://www.linuxdot.net/bbsfile-3084 一.Jexus简介:Jexus web server for linux 是一款基于.NET兼容环境,运行于Linux ...

  6. MySQL Explain详解(转)

    explain SELECT a.* FROM test a,(select id from test where level_id <=4 order by aa_id limit 24300 ...

  7. css3 利用perspective实现翻页效果和正方体 以及翻转效果

    要点: 1 实现3D效果就需要使用perspective属性 1 页面旋转使用css3的rorate 2 使用backface-visibility 实现正面元素翻转之后背面不可见,显示出反面的元素 ...

  8. Asp.net 实现只能允许一个账号同时只能在一个地方登录

    先上帮助类: /// <summary> /// 单点登录帮助类 /// </summary> public class SSOHelper { /// <summary ...

  9. struts2 ajax jquery返回json类型

    三个页面, <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE struts PUBLIC & ...

  10. play 1.2.4的action执行前后的加载逻辑

    ——杂言:最近在整理这一年的笔记,打算慢慢温故一遍,然后搬到博客里头来.这篇是2013.06.18时候整理的关于action执行前后的一些载入先后顺序逻辑.我的理解可能有偏差,如果有错误的,请读者们及 ...