[传送门]
题意
一直角坐标系中,BaoBao在原点位置,平面上有n个发射装置,任意两个发射器之间都有激光束,共(n-1)*n/2条激光,BaoBao的朋友可以去除任何发射器,问最少去除多少发射器,能使BaoBao不会被激光所伤还能从
(
0
,
0
)
(0,0)
(0,0)逃逸到
(
1
0
1
0
1
0
1
0
10
,
1
0
1
0
1
0
1
0
10
)
(10^{10^{10^{10^{10}}}},10^{10^{10^{10^{10}}}})
(1010101010,1010101010)
tip.发射器不会重合,也不会存在激光束穿过
(
0
,
0
)
(0,0)
(0,0)
思路&解法
首先需要知道计算计算几何的基础知识:极角排序、向量叉积及其意义
可以发现,在枚举一条过原点的直线,直线一侧的发射器都移除时,BaoBao才可以逃逸。但是直线太多,我们怎么处理呢。
如果学过计算几何的旋转卡壳,都能比较快的想到双指针枚举直线
先将所有点对原点极角排序,但是对于angel相同的需要去重加权,然后双指针指向枚举的直线一侧的角度最小的点和角度最大的点,双指针不断滑动表示枚举直线,枚举同时维护ans。
设
p
0
p_0
p0?位原点,枚举
p
l
p_l
pl?,寻找对应
p
r
p_r
pr?,当
p
0
p
l
?
\vec{p_0p_l}
p0?pl?
?与
p
0
p
r
?
\vec{p_0p_r}
p0?pr?
?的叉积 和
p
0
p
l
?
\vec{p_0p_l}
p0?pl?
?与
p
0
p
r
+
1
?
\vec{p_0p_{r+1}}
p0?pr+1?
?的叉积 异号,表示
p
l
p_l
pl?到
p
r
p_r
pr?都是直线一侧的点。
tips.由于坐标达到
∣
x
∣
,
∣
y
∣
≤
1
0
9
|x|,|y| \leq 10^9
∣x∣,∣y∣≤109,所以精度要求就是1e-18(小数点后18位)(大概是这样),所以储存angel时double的精度是不够的,要用到long double,但是计算过程中大量涉及叉积,坐标用long double计算速度就慢,会T,所以坐标的储存推荐使用long long(格点,且不推荐int,计算叉积过程中会爆int)
Accepted code
#include <bits/stdc++.h>
using namespace std;
#define db long double
#define maxn 1000005
#define ll long long
#define Vector point
#define id(x) ((x - 1) % (len - 1) + 1)
const db eps = 1e-18;
int _, n, len = 1;
struct point {
ll x, y;
db ag;
point() {}
point(ll xx, ll yy) : x(xx), y(yy) { ag = atan2((db)y, (db)x); }
point operator-(const point &a) { return point(x - a.x, y - a.y); }
bool operator<(const point &a) const {
return ag < a.ag;
}
};
point p[maxn], pp[maxn];
int num[maxn];
db Cross(Vector a, Vector b) { return a.x * b.y - a.y * b.x; }
db Dot(Vector a, Vector b) { return a.x * b.x + a.y * b.y; }
point p0(0, 0);
int ans;
bool sgn(db a, db b) { return fabs(a - b) < eps; }
bool sgn(point a, point b) {
a = a - p0, b = b - p0;
return fabs(a.ag - b.ag) < eps;
}
bool ck(int l, int r) {
return Cross(pp[l] - p0, pp[r] - p0) * Cross(pp[l] - p0, pp[id(r + 1)] - p0) < eps
&& !sgn(pp[l], pp[r]) && !sgn(pp[l], pp[id(r + 1)]);
}
int main() {
for (scanf("%d", &_); _; _--) {
len = 1;
scanf("%d", &n);
ans = n;
for (int i = 0, x, y; i < n; i++) {
scanf("%d%d", &x, &y);
p[i] = point(x, y);
}
sort(p, p + n);
for (int i = 0; i < n; i++) {
if (i && sgn(p[i], p[i - 1])) {
++num[len - 1];
} else {
pp[len] = p[i];
num[len++] = 1;
}
}
if (len - 1 < 3) {
printf("0\n");
continue;
}
for (int i = 1; i < len; i++) num[i] += num[i - 1];
ll r = 1, l = 1;
for (; r < len; r++)
if (ck(l, r) || id(r + 1) == id(l)) break;
while (l < len) {
for (;; ++r)
if (ck(id(l), id(r)) || id(r + 1) == id(l)) break;
if (id(l) <= id(r)) ans = min(ans, min(n - (num[id(r)] - num[id(l) - 1]), num[id(r)] - num[id(l) - 1]));
else ans = min(ans, min(num[id(r)] + n - num[id(l) - 1], n - (num[id(r)] + n - num[id(l) - 1])));
++l;
}
printf("%d\n", ans);
}
return 0;
}
|