水平可见直线 bzoj 1007

水平可见直线 (1s 128M) lines

【问题描述】

在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条直线两两不重合.求出所有可见的直线.

【输入格式】

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

【输出格式】

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

【样例输入】

3

-1 0

1 0

0 0

【样例输出】

1 2



题解:

主要算法:计算几何;快速排序;单调栈;

1.对于斜率相同的两条直线截距小的被覆盖。

2.对于斜率不同的三条直线,如果一条直线不可见

那么必定是斜率最大和斜率最小的覆盖另外一条线段

同时斜率最大和斜率最小的直线的交点在另一条线段的上方

根据这个性质,通过排序和单调栈即可维护可见直线。

 1 #include<algorithm>
 2 #include<iostream>
 3 #include<cstring>
 4 #include<cstdlib>
 5 #include<cstdio>
 6 #include<cmath>
 7 using namespace std;
 8 inline int Get()
 9 {
10     int x = 0, s = 1;
11     char c = getchar();
12     while(‘0‘ > c || c > ‘9‘)
13     {
14         if(c == ‘-‘) s = -1;
15         c = getchar();
16     }
17     while(‘0‘ <= c && c <= ‘9‘)
18     {
19         x = (x << 3) + (x << 1) + c - ‘0‘;
20         c = getchar();
21     }
22     return x * s;
23 }
24 int n;
25 struct shape
26 {
27     int a, b, i;
28 };
29 shape a[100233];
30 int s[100233];
31 int ans[100233];
32 inline bool rule(shape a, shape b)
33 {
34     if(a.a != b.a) return a.a > b.a;
35     return a.b > b.b;
36 }
37 inline double Sol(int x, int y)
38 {
39     return (double) (a[y].b - a[x].b) / (double) (a[x].a - a[y].a);
40 }
41 int main()
42 {
43     n = Get();
44     for(int i = 1; i <= n; ++i)
45     {
46         a[i].a = Get();
47         a[i].b = Get();
48         a[i].i = i;
49     }
50     sort(a + 1, a + 1 + n, rule);
51     int top = 0;
52     for(int i = 1; i <= n; ++i)
53     {
54         if(a[i].a == a[s[top]].a) continue;
55         while(top > 1 && Sol(s[top], i) >= Sol(s[top], s[top - 1]))
56             --top;
57         s[++top] = i;
58         ans[top] = a[i].i;
59     }
60     sort(ans + 1, ans + 1 + top);
61     for(int i = 1; i <= top; ++i) printf("%d ", ans[i]);
62 }
时间: 2024-10-19 07:41:35

水平可见直线 bzoj 1007的相关文章

AC日记——[HNOI2008]水平可见直线 bzoj 1007

1007 思路: 维护一个下凸壳: 用单调栈来维护这玩意儿: 先将斜率排序: 然后判断栈顶元素和当前元素的交点x是否小于栈顶元素和栈顶上一个元素的交点x: 注意: 人神共愤的精度问题和输出空格问题: 来,上代码: #include <cmath> #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> using namespace std; #d

bzoj 1007 水平可见直线 半平面交稀里糊涂的过了...

题意:按y=Ax+B的形式给出n(<=50000)条直线,求从y值为无穷大的地方向下看能看到的直线编号 一看到题目就想到半平面交,以每条直线的上方为一个半平面,求半平面的交,交集中存在的直线就是能看到的直线 但是写出来之后发现样例都过不了... 对于样例,如果允许半平面在边界处重叠那么答案是1,2,3,如果不允许只有1.. 然后抱着试一试的心理交上去了,结果竟然直接AC了.. 后来看题解只需要考虑交点x坐标.. bzoj 1007 水平可见直线 半平面交稀里糊涂的过了...,布布扣,bubuko

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

计算几何初步 其实是维护一个类似下凸壳的东西?画图后发现其实斜率是单调递增的,交点的横坐标也是单调递增的,所以排序一下搞个单调栈来做就可以了…… 看了hzwer的做法…… 1 /************************************************************** 2 Problem: 1007 3 User: Tunix 4 Language: C++ 5 Result: Accepted 6 Time:252 ms 7 Memory:2812 kb 8

BZOJ 1007 水平可见直线 | 计算几何

BZOJ 1007 水平可见直线 题面 平面直角坐标系上有一些直线,请求出在纵坐标无限大处能看到哪些直线. 题解 将所有直线按照斜率排序(平行的直线只保留最高的直线),维护一个栈,当当前直线与栈顶直线的交点在栈顶两条直线的交点的左边,则弹出栈顶元素.可以画图证明这是正确的(因为我们要维护一个下凸的图形). #include <cmath> #include <cstdio> #include <cstring> #include <algorithm> us

bzoj 1007 水平可见直线

bzoj 1007 水平可见直线 这里的半平面都是 \(y\geq kx+b\) 类型的,将直线 \(l:y=kx+b\) 对应到点 \((k,-b)\) ,转化成凸包求解即可. 如果有两种类型,需分类后分别求上下凸包,最后去重,合并. #include<bits/stdc++.h> using namespace std; #define ll long long #define mp make_pair #define pii pair<int,int> inline int

BZOJ 1007: [HNOI2008]水平可见直线( 计算几何 )

按A从小到大排序然后用栈解决. -------------------------------------------------------------------- #include<bits/stdc++.h> using namespace std; typedef long long ll; const int maxn = 50009; struct L { int A, B, id; inline void Read(int p) { scanf("%d%d"

bzoj 1007 [HNOI2008]水平可见直线(单调栈)

1007: [HNOI2008]水平可见直线 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5120  Solved: 1899[Submit][Status][Discuss] Description 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则Li为被覆盖的.     例如,对于直线:     L1:y=x; L2:y=-x; L3:y=0     则L1和L

【凸壳】【HNOI 2008】【bzoj 1007】水平可见直线

1007: [HNOI2008]水平可见直线 Time Limit: 1 Sec Memory Limit: 162 MB Submit: 4567 Solved: 1686 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|,|

BZOJ 1007: [HNOI2008]水平可见直线 栈/计算几何

1007: [HNOI2008]水平可见直线 Time Limit: 1 Sec  Memory Limit: 162 MB 题目连接 http://www.lydsy.com/JudgeOnline/problem.php?id=1007 Description 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则Li为被覆盖的.     例如,对于直线:     L1:y=x; L2:y=-x; L3:y=0