要点

  • 用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. 删除老的Azure Blob Snapshot

    客户有这样的需求:每天需要对VM的数据进行备份,但如果备份的时间超过一定的天数,需要进行清除. 本文也是在前一篇Azure Blob Snapshot上的优化. "Azure blob St ...

  2. POJ3126Prime Path(BFS)

    #include"cstdio" #include"queue" #include"cstring" using namespace std ...

  3. strdup与strndup

    strdup()函数是c语言中常用的一种字符串拷贝库函数,一般和free()函数成对出现. extern char *strdup(char *s); 头文件:string.h 功 能: 将串拷贝到新 ...

  4. 自定义滚动条jQuery插件- Perfect Scrollbar

    主要特性: 不需要修改任何的元素的css 滚动条不影响最初的页面布局设计 滚动条支持完整的自定义 滚动条的尺寸和位置会随着容器尺寸或者内容的变化而变化 依赖于jQuery和相关几个类库 不需要定义宽度 ...

  5. 12.Redis Select 命令 - 切换到指定的数据库

    转自:http://www.runoob.com/redis/redis-tutorial.html Redis Select 命令用于切换到指定的数据库,数据库索引号 index 用数字值指定,以 ...

  6. js中object、字符串与正则表达式的方法

    对象 1.object.hasOwnProperty(name) 检测object是否包含一个名为name的属性,那么hasOwnProperty方法返回true,但是不包括其原型上的属性. 正则表达 ...

  7. [hdu4372]counting buildings

    解题关键: n的环排列的个数与n-1个元素的排列的个数相等. 首先可以肯定,无论从最左边还是从最右边看,最高的那个楼一定是可以看到的,从这里入手. 假设最高的楼的位置固定,最高楼的编号为n,那么我们为 ...

  8. 《精通Spring4.X企业应用开发实战》读后感第六章(内部工作机制、BeanDefinition、InstantiationStrategy、BeanWrapper)

  9. tensorflow第一个例子

    import tensorflow as tf import numpy as np # create data x_data = np.random.rand(100).astype(np.floa ...

  10. .NET中的泛型委托

    .Net中有一个内置的委托 Func 它总共有以下5种形式 1.  Func<TResult> 2.  Func<T,TResult> 3.  Func<T1,T2,TR ...