POJ 3304 Segments (直线与线段是否相交)

时间:2023-03-09 16:20:06
POJ 3304 Segments (直线与线段是否相交)

题目链接

题意 : 能否找出一条直线使得所有给定的线段在该直线上的投影有一个公共点。

思路 : 假设存在一条直线a使得所有线段在该直线上的投影有公共点,则必存在一条垂直于直线a的直线b,直线b与所有线段相交,所以问题又转变为是否存在一条直线与所在所有线段相交。

假设这样的直线存在,则这一条直线可能与某一条或者某些线段的端点重合,也可能不重合。对于那些没有在端点相交的线段,我们可以把这一条直线通过旋转或平移,让其先与一条线段在线段的端点相交(那此时这一条直线与别的线段就在别的线段的中间相交), 然后继续旋转,让这一条直线与别的直线也在端点处相交。到此为止,我们可以明白所有存在的这样的直线都可以这样平移和旋转来处理他。那么反过来,如果我们用所有的线段的端点(任意组合)构成的直线都不满足和所有的线段至少有一个公共点,也就是说不存在这样的直线。

要注意的是小于1e-8就算是重点,所以要注意判断

 #include <cstdio>
#include <cstring>
#include <iostream>
#include <cmath>
#include <algorithm>
#define eps 1e-8 using namespace std ; struct point
{
double x,y;
}p[];
struct line
{
point a,b;
}L[]; double multi(point a,point b,point c)
{
return ((a.x-c.x)*(b.y-c.y) - (b.x-c.x)*(a.y-c.y)) ;
}
bool inter(line L,point a,point b)
{
double x1 = multi(L.a,a,b) ;
double x2 = multi(L.b,a,b) ;
if((x1 > eps && x2 < -eps) || (x1 < -eps && x2 > eps) || (fabs(x1) < eps) || (fabs(x2) < eps))
return true ;
return false ;
}
int main()
{
int T ,n;
scanf("%d",&T) ;
while(T--)
{
scanf("%d",&n) ;
int cnt = ;
for(int i = ; i < n ; i++)
{
scanf("%lf %lf %lf %lf",&L[i].a.x,&L[i].a.y,&L[i].b.x,&L[i].b.y) ;
p[cnt ++] = L[i].a ;
p[cnt ++] = L[i].b ;
// printf("2\n") ;
}
//printf("1\n") ;
bool ans = false ;
for(int i = ; i < cnt- ; i++)
{
for(int j = i+ ; j < cnt ; j++)
{
if(fabs(p[i].x-p[j].x) < eps && fabs(p[i].y-p[j].y) < eps) continue ;
bool flag = true ;
for(int k = ; k < n ; k++)
{
if(!inter(L[k],p[i],p[j]))
{
flag = false ;
break ;
}
}
if(flag)
{
ans = true ;
break ;
}
}
}
if(ans) puts("Yes!") ;
else puts("No!") ;
}
return ;
}