[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
 
 
解题思路:
  这是一道几何水题,算法很简单,先按照斜率排序,依次处理每一条直线
联立 y-kx1-b1=0 
   y-kx2-b2=0
求得两直线焦点横坐标,我们会发现由直线组成的凸包随K值递增,交点横左边也是递增的
所以用一个单调栈来维护直线集合,如果 交点(k,j)在交点(i,j)左边,则直线i一定被覆盖
所以将直线i 弹出栈,以此类推。
 
 
AC代码:
 

#include<cstdio>
#include<stack>
#include<algorithm>
#define eps 1e-8
#define Maxn 50001
using namespace std;
typedef double D;
class Node{
public:
int num;
D k,b;
};
stack<Node>Q;
Node t[Maxn],h[Maxn];
bool cmp(const Node A,const Node B){
return (A.k<B.k)||((A.k-B.k<eps)&&(A.b>B.b)); //problem 1
}
bool cmp2(const Node A,const Node B){
return A.num<B.num;
}
double public_node(Node A,Node B){
return (B.b-A.b)/(A.k-B.k);
}
bool Istrue(int i){
Node tmp=Q.top(),tmp2;
Q.pop();
if(!Q.empty())
tmp2=Q.top();
else
tmp2=tmp;
Q.push(tmp);
if(public_node(h[i],tmp2)<=public_node(tmp,tmp2))
return true;
return false;
}
int main(){
int n;
scanf("%d",&n);
for(int i=0;i<n;++i)
scanf("%lf%lf",&h[i].k,&h[i].b),
h[i].num=i;
sort(h,h+n,cmp);
Q.push(h[0]);
for(int i=1;i<n;++i){
if((h[i].k-h[i-1].k)<eps)    // problem 2
continue;
while(Q.size()>1&&!Q.empty()&&Istrue(i))
Q.pop();
Q.push(h[i]);
}
int i=0;
while(!Q.empty()){
t[i++]=Q.top();
Q.pop();
}
sort(t,t+i,cmp2);
for(int j=0;j<i;++j)
printf("%d ",t[j].num+1);
}

 
注意情况 :
代码中注释的 P1 P2,切记不要将 Node 中的k,b写为整形 ,若写为浮点形则 逻辑关系上不可以使用等于
常识性的知识: 两个浮点数比较大小只能以做差的方式来进行比较。

转载于:https://www.cnblogs.com/PengGG/p/4822240.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值