bzoj2961 共点圆

Posted MashiroSky

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了bzoj2961 共点圆相关的知识,希望对你有一定的参考价值。

http://www.lydsy.com/JudgeOnline/problem.php?id=2961 (题目链接)

题意

  按照一定的顺序给出一些圆和一些点,对于每一个点问是否在所有圆内。

Solution

  我算是明白计算几何题是有多蛋疼了。

  圆包含点$(x_0,y_0)$的条件:$$x*x+y*y>=(x-x_0)*(x-x_0)+(y-y_0)*(y-y_0)$$

$$-2x_0+x_0^2+y_0^2<=2y_0y$$

  题目只说圆心的纵坐标大于$0$,气的我吐出一口老血。所以根据$y_0$的正负,分类讨论,每种情况都是一个半平面,直线的斜率为$-x_0/y_0$,然后我们维护一个上凸包和一个下凸包对询问进行更新,CDQ分治求解就可以了。

细节

  各种细节蛋疼死了,横坐标相等斜率特判,而且是有向线段的斜率两个点的顺序不能乱写。一开始还把斜截式写成了一般式搞了半天我就说怎么不对。

代码

// bzoj2961
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define LD long double
#define inf 1e40
#define eps 1e-10
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout)
using namespace std;

const int maxn=500010;
int n,qu[maxn],st[maxn],ans[maxn];
struct data {LD x,y,k;int op,id;}q[maxn],nq[maxn];

bool cmpk(data a,data b) {return a.k<b.k;}
bool cmpid(data a,data b) {return a.id<b.id;}
LD slope(data a,data b) {
	return fabs(a.x-b.x)<=eps ? inf*(a.y<b.y ? 1 : -1) : (b.y-a.y)/(b.x-a.x);  //mdzz一定要写a.y<b.y而不是a.y>b.y,有向线段
}
LD dis(int x,int y) {
	return (q[x].x-q[y].x)*(q[x].x-q[y].x)+(q[x].y-q[y].y)*(q[x].y-q[y].y);
}
void solve(int l,int r) {
	if (l==r) return;
	int mid=(l+r)>>1,l1=l,l2=mid+1,top=0,h=1,t=0;
	for (int i=l;i<=r;i++) q[i].id<=mid ? nq[l1++]=q[i] : nq[l2++]=q[i];
	for (int i=l;i<=r;i++) q[i]=nq[i];
	solve(l,mid);
	for (int i=l;i<=mid;i++) if (!q[i].op) {
			while (top>1 && slope(q[st[top-1]],q[st[top]])<slope(q[st[top]],q[i])+eps) top--;
			st[++top]=i;
			while (h<t && slope(q[qu[t-1]],q[qu[t]])>slope(q[qu[t]],q[i])-eps) t--;
			qu[++t]=i;
		}
	for (int i=mid+1;i<=r;i++) if (q[i].op) {
			if (q[i].y<0) {
				while (top>1 && slope(q[st[top-1]],q[st[top]])<q[i].k) top--;
				if (top) ans[q[i].id]&=(dis(i,st[top])<=dis(st[top],0)+eps);   //判断条件一定要加
			}
			else {
				while (h<t && slope(q[qu[h]],q[qu[h+1]])<q[i].k) h++;
				if (h<=t) ans[q[i].id]&=(dis(i,qu[h])<=dis(qu[h],0)+eps);   //判断条件一定要加
			}
		}
	solve(mid+1,r);
	for (int i=l,j=mid+1,k=l;i<=mid || j<=r;) {
		if (j>r || (i<=mid && q[i].x<q[j].x)) nq[k++]=q[i++];
		else nq[k++]=q[j++];
	}
	for (int i=l;i<=r;i++) q[i]=nq[i];
}
int main() {
	scanf("%d",&n);
	for (int i=1;i<=n;i++) {
		scanf("%d%Lf%Lf",&q[i].op,&q[i].x,&q[i].y);
		if (q[i].op) q[i].k=-q[i].x/q[i].y;
		ans[i]=1;q[i].id=i;
	}
	sort(q+1,q+1+n,cmpk);
	solve(1,n);
	sort(q+1,q+1+n,cmpid);
	for (int flag=0,i=1;i<=n;i++) {
		if (q[i].op) puts(flag && ans[q[i].id] ? "Yes" : "No");
		else flag=1;
	}
	return 0;
}

 

以上是关于bzoj2961 共点圆的主要内容,如果未能解决你的问题,请参考以下文章

bzoj千题计划176:bzoj1199: [HNOI2005]汤姆的游戏

打卡 打印点圆圆柱信息

BZOJ-1192: [HNOI2006]鬼谷子的钱袋

前端学习(2961):前一天回顾

前端学习(2961):前一天回顾

POJ2961_kmp