Codeforces 350D(计算几何)
要点
- 用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(计算几何)的更多相关文章
- Codeforces Gym100543B 计算几何 凸包 线段树 二分/三分 卡常
原文链接https://www.cnblogs.com/zhouzhendong/p/CF-Gym100543B.html 题目传送门 - CF-Gym100543B 题意 给定一个折线图,对于每一条 ...
- 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 ...
- Codeforces 749B:Parallelogram is Back(计算几何)
http://codeforces.com/problemset/problem/749/B 题意:已知平行四边形三个顶点,求另外一个顶点可能的位置. 思路:用向量来做. #include <c ...
- 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 ...
- Codeforces Round #524 (Div. 2) C. Masha and two friends(思维+计算几何?)
传送门 https://www.cnblogs.com/violet-acmer/p/10146350.html 题意: 有一块 n*m 的棋盘,初始,黑白块相间排列,且左下角为白块. 给出两个区间[ ...
- Codeforces 528E Triangles 3000 - 计算几何
题目传送门 传送点I 传送点II 传送点III 题目大意 给定$n$的平面上的直线,保证没有三条直线共点,两条直线平行.问随机选出3条直线交成的三角形面积的期望. 显然$S=\frac{1}{2}ah ...
- 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 ...
- 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 ...
- Codeforces Beta Round #1 C. Ancient Berland Circus 计算几何
C. Ancient Berland Circus 题目连接: http://www.codeforces.com/contest/1/problem/C Description Nowadays a ...
随机推荐
- 删除老的Azure Blob Snapshot
客户有这样的需求:每天需要对VM的数据进行备份,但如果备份的时间超过一定的天数,需要进行清除. 本文也是在前一篇Azure Blob Snapshot上的优化. "Azure blob St ...
- POJ3126Prime Path(BFS)
#include"cstdio" #include"queue" #include"cstring" using namespace std ...
- strdup与strndup
strdup()函数是c语言中常用的一种字符串拷贝库函数,一般和free()函数成对出现. extern char *strdup(char *s); 头文件:string.h 功 能: 将串拷贝到新 ...
- 自定义滚动条jQuery插件- Perfect Scrollbar
主要特性: 不需要修改任何的元素的css 滚动条不影响最初的页面布局设计 滚动条支持完整的自定义 滚动条的尺寸和位置会随着容器尺寸或者内容的变化而变化 依赖于jQuery和相关几个类库 不需要定义宽度 ...
- 12.Redis Select 命令 - 切换到指定的数据库
转自:http://www.runoob.com/redis/redis-tutorial.html Redis Select 命令用于切换到指定的数据库,数据库索引号 index 用数字值指定,以 ...
- js中object、字符串与正则表达式的方法
对象 1.object.hasOwnProperty(name) 检测object是否包含一个名为name的属性,那么hasOwnProperty方法返回true,但是不包括其原型上的属性. 正则表达 ...
- [hdu4372]counting buildings
解题关键: n的环排列的个数与n-1个元素的排列的个数相等. 首先可以肯定,无论从最左边还是从最右边看,最高的那个楼一定是可以看到的,从这里入手. 假设最高的楼的位置固定,最高楼的编号为n,那么我们为 ...
- 《精通Spring4.X企业应用开发实战》读后感第六章(内部工作机制、BeanDefinition、InstantiationStrategy、BeanWrapper)
- tensorflow第一个例子
import tensorflow as tf import numpy as np # create data x_data = np.random.rand(100).astype(np.floa ...
- .NET中的泛型委托
.Net中有一个内置的委托 Func 它总共有以下5种形式 1. Func<TResult> 2. Func<T,TResult> 3. Func<T1,T2,TR ...