C - Segments POJ - 3304 (判断线段相交)

时间:2024-07-26 08:33:38

题目链接:https://vjudge.net/contest/276358#problem/C

题目大意:给你n条线段,问你是否存在一条线段使得所有的线段在这条直线的投影至少具有一个交点?

具体思路:这个题转换一下思路,假设存在一条直线与所有的线段都相交,那么这条直线的垂线就是题目中所求的直线,我们可以把所有的端点都遍历一遍,询问一下看有没有符合条件的直线就可以了。

AC代码:

 /*************************************************************************
> File Name: hqx.cpp
> Author: ma6174
> Mail: ma6174@163.com
> Created Time: 2019年01月28日 星期一 00时24分52秒
************************************************************************/
#include<iostream>
#include<stack>
#include<stdio.h>
#include<iomanip>
#include<cmath>
using namespace std;
# define ll long long
const int maxn = +;
const double eps = 1e-;
struct node{
double x1,y1;
double x2,y2;
}q[maxn];
int n;
double dis(double x1,double y1,double x2,double y2){
return sqrt((x2-x1)*(x2-x1)+(y2-y1)*(y2-y1));
}
double cal(double x1,double y1,double x2,double y2,double x,double y){
return (x2-x1)*(y-y1)-(x-x1)*(y2-y1);//叉积
}
bool check(double x1,double y1,double x2,double y2){
if(dis(x1,y1,x2,y2)<eps)return false;//构不成线段
for(int i=;i<=n;i++){
if(cal(x1,y1,x2,y2,q[i].x1,q[i].y1)*cal(x1,y1,x2,y2,q[i].x2,q[i].y2)>eps)return false;//判段线段是不是相交
}
return true;
}
int main( ) {
int T;
scanf("%d",&T);
while(T--){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%lf %lf %lf %lf",&q[i].x1,&q[i].y1,&q[i].x2,&q[i].y2);
}
int flag=;
if(n==)flag=;
else {
for(int i=;i<=n;i++){
for(int j=i+;j<=n;j++){
if(check(q[i].x1,q[i].y1,q[j].x1,q[j].y1)||check(q[i].x1,q[i].y1,q[j].x2,q[j].y2)||check(q[i].x2,q[i].y2,q[j].x1,q[j].y1)||
check(q[i].x2,q[i].y2,q[j].x2,q[j].y2)){ flag=;
break;
}
}
if(!flag)break;
}
}
if(!flag)printf("Yes!\n");
else printf("No!\n");
}
return ;
}