Line of Sight 计算几何基础

题意:

题目链接
在房屋与property line之间有障碍物(房屋,property line,障碍物均可看做与x轴平行的线段)
求从房屋到property line最长的能看到的一段的长度

思路:

能看到的一段的长度本身并不好求解
但是不能看到的一段的长度却相对好求解
于是找到property line上看不见的每一段
而后按照左端点排序,乱搞即可

注意事项:

  • 障碍物不一定在property line与房屋之间
  • 可能没有障碍物
  • 边界情况

code:

#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=5e4+5;
const double eps=1e-7;
struct sec{double l,r;}a[N];
bool operator<(sec x,sec y){return x.l<y.l;}
struct point{double x,y;};
point e1=(point){0,0},e2=(point){50,0};
double ans;
point operator-(point x,point y){return (point){x.x-y.x,x.y-y.y};}
double operator^(point x,point y){return x.x*y.y-x.y*y.x;}
struct seg{point u,v;}hs,pro;
inline double inc(point p1,point p2)
{
	double s1=(e1-p1)^(e2-p1),s2=(e2-p2)^(e1-p2);
	return p2.x+(p1.x-p2.x)*s2/(s1+s2);
}
int main()
{
	while(7)
	{
		ans=0;
		scanf("%lf%lf%lf",&hs.u.x,&hs.v.x,&hs.u.y);
		hs.v.y=hs.u.y;
		if(hs.u.x<eps&&hs.v.x<eps&&hs.u.y<eps)break;
		scanf("%lf%lf%lf",&pro.u.x,&pro.v.x,&pro.u.y);
		pro.v.y=pro.u.y;e1.y=e2.y=pro.u.y;
		int num=0,cnt=0;scanf("%d",&num);
		for(int i=1;i<=num;++i)
		{
			double x1=0,x2=0,y=0;
			scanf("%lf%lf%lf",&x1,&x2,&y);
			if(y-hs.u.y>-eps||pro.u.y-y>-eps)continue;
			a[++cnt].l=inc(hs.u,(point){x2,y});
			a[cnt].l=min(pro.v.x,max(pro.u.x,a[cnt].l));
			a[cnt].r=inc(hs.v,(point){x1,y});
			a[cnt].r=min(pro.v.x,max(pro.u.x,a[cnt].r));
			swap(a[cnt].l,a[cnt].r);
		}
		if(!cnt){printf("%.2f\n",pro.v.x-pro.u.x);continue;}
		sort(a+1,a+cnt+1);
		double ll=-1,rr=-1;
		for(int i=1;i<=cnt;++i)
		{
			if(ll<-eps&&rr<-eps)
			{
				ll=a[i].l,rr=a[i].r;
				ans=max(ans,ll-pro.u.x);
				continue;
			}
			if(rr-a[i].l>-eps)rr=max(rr,a[i].r);
			else
			{
				ans=max(ans,a[i].l-rr);
				ll=a[i].l,rr=a[i].r;
			}
		}
		ans=max(ans,pro.v.x-rr);
		if(ans<eps)puts("No View");
		else printf("%.2f\n",ans);
	}
	return 0;
}
posted @ 2020-01-21 10:30  BILL666  阅读(270)  评论(0编辑  收藏  举报