[传送门]
一直角坐标系中,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} p0pl与 p 0 p r ⃗ \vec{p_0p_r} p0pr的叉积 和 p 0 p l ⃗ \vec{p_0p_l} p0pl与 p 0 p r + 1 ⃗ \vec{p_0p_{r+1}} p0pr+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)
#include
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 atan2(y, x) < atan2(a.y, a.x);
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;
}