【BZOJ 1007】 [HNOI2008]水平可见直线

Description

 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则Li为被覆盖的.
    例如,对于直线:
    L1:y=x; L2:y=-x; L3:y=0
    则L1和L2是可见的,L3是被覆盖的.
    给出n条直线,表示成y=Ax+B的形式(|A|,|B|<=500000),且n条直线两两不重合.求出所有可见的直线.

Input

第一行为N(0 < N < 50000),接下来的N行输入Ai,Bi

Output

从小到大输出可见直线的编号,两两中间用空格隔开,最后一个数字后面也必须有个空格

Sample Input

3
-1 0
1 0
0 0

Sample Output

1 2
 
水平可见直线一定构成一个下凸的半凸包,画画图就能发现,这个下凸包的交点的横坐标单调递增,这样就可以用一个栈来维护
然后就没有了
 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<iostream>
 4 #include<cmath>
 5 #define eps 1e-8
 6 const int inf=10000000;
 7 using namespace std;
 8 struct node{double a,b;int no;}l[100010];
 9 int n,top,stack[100100];
10 double x;
11 bool cmp(node a,node b){
12     if(fabs(a.a-b.a)<eps)return a.b<b.b;
13     else return a.a<b.a;
14 }
15  
16 double cal(node a,node b){
17     return (a.b-b.b)/(b.a-a.a);
18 }
19  
20 bool cmp2(int a,int b){
21     return l[a].no<l[b].no;
22 }
23  
24 int main(){
25     scanf("%d",&n);
26     for(int i=1;i<=n;i++)
27         scanf("%lf%lf",&l[i].a,&l[i].b),l[i].no=i;
28     sort(l+1,l+n+1,cmp);
29     double now=(l[1].b-l[2].b)/(l[2].a-l[1].a);
30     for(int i=1;i<=n;i++){
31         while(top){
32             if(fabs(l[i].a-l[stack[top]].a)<eps) top--;
33             else if(top>1&&cal(l[i],l[stack[top-1]])<=cal(l[stack[top-1]],l[stack[top]]))
34                 top--;
35                 else break;
36         }
37         stack[++top]=i;
38     }
39     sort(stack+1,stack+1+top,cmp2);
40     printf("%d",l[stack[1]].no);
41     for(int i=2;i<=top;i++) printf(" %d",l[stack[i]].no);
42 }
原文地址:https://www.cnblogs.com/wuminyan/p/5215665.html