[扫描线]POJ2932 Coneology
题意:有n个圆 依次给了半径和圆心坐标 保证输入的圆不相交(只有 相离 和 内含/外含 的情况)
问 有几个圆 不内含在其他圆中,并分别列出这几个圆的编号(1~n)
(n的范围是[1, 40000])
案例画出来大概是这样的
(那个原点为(50,50)的太远了,就意思一下)
所以答案是3号圆和5号圆 不被包含
好了,若这道题n只有1000,那么只要for两层,每个圆与另外的圆比较, 判断圆心是否在其他圆内即可判断是否包含
这样的复杂度是O($n^2$)
可是现在n有40000,显然不能用O($n^2$)来解决
由这道题,简单粗暴的学习了一下扫描线, 复杂度为O(nlogn)
什么是扫描线呢?
顾名思义,就是一根线,扫描过去。
怎样的一根线,怎么扫过去呢?
①平行于x轴,自上而下/自下而上 扫描
②平行于y轴,自左而右/自右而左 扫描
③绕圆心 逆时针/顺时针 扫描
扫描线要干什么呢?
“扫描线在平面上按给定轨迹移动的同时,不断根据扫描线扫过部分更新信息,从而得到整体所要求的结果”
这道题,可以用上述的①/②
自左而右扫描时,只有当扫描线移动到圆的左右两端时,线与圆的关系才会发生改变,因此先把圆的左右端点取出来排个序
每当遇到某圆的左端点,判断该圆是否包含在其他圆内
因为所有的圆都不相交,因此,每个圆都只可能包含在上下两个与它最相近的圆中
(此处“上下两个”是通过比较 圆心的纵坐标 来确定的)
也就是 每当我们得到一个不包含在其他圆中的圆,我们需要将它存起来,并将这些圆按圆心的纵坐标排序 以便与下一个扫到的圆进行比较
我们可以用set<pair<double, int> >来存 (pair.first是圆心的纵坐标,pair.second是圆的编号) set会自动按pair.first排序
当我们扫到某圆的右端点时,表示该圆的影响范围已经扫完了,后面扫到的圆不可能包含在该圆中,因此可以把该圆从set中去掉
查找“上下两个与它最近的圆”的复杂度为O(logn)
因此遍历n个圆的复杂度为O(nlogn)
const int N=; double x[N], y[N] ,r[N]; bool inside(int i, int j) // i是否在j内
{
double dx=x[i]-x[j], dy=y[i]-y[j];
return dx*dx+dy*dy<=r[j]*r[j];
} int main()
{
int n;
while(~scanf("%d", &n))
{
for(int i=;i<n;i++)
scanf("%lf%lf%lf", &r[i], &x[i], &y[i]);
vector<pair<double, int> > X;
for(int i=;i<n;i++)
{
X.push_back(make_pair(x[i]-r[i], i)); // 左
X.push_back(make_pair(x[i]+r[i], i+n));// 右
}
sort(X.begin(), X.end());
set<pair<double, int> > out;
vector<int> ans;
for(int i=;i<X.size();i++) // 从左到右扫描
{
int id=X[i].second%n;
if(X[i].second<n) // 左
{
set<pair<double, int> >::iterator it=out.lower_bound(make_pair(y[id], id));
if(it!=out.end() && inside(id, it->second)) // id 在 前一个圆内 不加入
continue;
if(it!=out.begin() && inside(id, (--it)->second)) // id 在 后一个圆内 不加入
continue;
ans.push_back(id);
out.insert(make_pair(y[id], id));
}
else // 右
out.erase(make_pair(y[id], id));
}
sort(ans.begin(), ans.end());
printf("%d\n", ans.size());
for(int i=;i<ans.size();i++)
printf("%d%c", ans[i]+, i+==ans.size()? '\n':' ');
}
return ;
}
POJ 2932
[扫描线]POJ2932 Coneology的更多相关文章
- POJ2932 Coneology【圆扫描线】
POJ2932 Coneology 题意: 给出一些不相交的圆,问有多少个圆不被其他圆包围 题解: 扫描线,把所有圆的左边界和右边界放到\(vector\)里排序,遍历到圆左边界的时候判断是否满足条件 ...
- poj2932 Coneology (扫描线)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud Coneology Time Limit: 5000MS Memory Lim ...
- 计算几何值平面扫面poj2932 Coneology
Coneology Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 4097 Accepted: 859 Descript ...
- poj2932 Coneology
地址:http://poj.org/problem?id=2932 题目: Coneology Time Limit: 5000MS Memory Limit: 65536K Total Subm ...
- 刷题总结——coneology(poj2932 扫描线)
题目: Description A student named Round Square loved to play with cones. He would arrange cones with d ...
- poj 2932 Coneology(扫描线+set)
Coneology Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 3574 Accepted: 680 Descript ...
- poj 2932 Coneology (扫描线)
题意 平面上有N个两两不相交的圆,求全部最外层的,即不被其它圆包括的圆的个数并输出 思路 挑战程序竞赛P259页 代码 /* ************************************* ...
- POJ 2932 Coneology(扫描线)
[题目链接] http://poj.org/problem?id=2932 [题目大意] 给出N个两两没有公共点的圆,求所有不包含于其它圆内部的圆 [题解] 我们计算出所有点在圆心所有y位置的x值, ...
- Coneology(POJ 2932)
原题如下: Coneology Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 4937 Accepted: 1086 D ...
随机推荐
- css笔记——关于 body/html 的高度百分比
body{ height:100%; 视窗的高度 min-height:100%;文档的具体高度} 这两个百分比的具体高度在页脚永远放在文档底部非常重要,此时用min-height:100% 具体 ...
- JavaScript、jQuery、HTML5、Node.js实例大全-读书笔记3
技术很多,例子很多,只好慢慢学,慢慢实践!!现在学的这本书是[JavaScript实战----JavaScript.jQuery.HTML5.Node.js实例大全] JavaScript.jQuer ...
- 【leetcode】10.Regular Expression Matching
题目描述: Implement regular expression matching with support for '.' and '*'. '.' Matches any single cha ...
- NOIP2011(提高组)DAY2---观光公交(vijosP1741)
描述 风景迷人的小城Y市,拥有n个美丽的景点.由于慕名而来的游客越来越多,Y市特意安排了一辆观光公交车,为游客提供更便捷的交通服务.观光公交车在第0分钟出现在1号景点,随后依次前往2.3.4……n号景 ...
- linux之gdb使用
gdb是linux下用来调试的一款软件,在这里,我只记录平常经常会用到的知识点,用到什么,就记录什么,在调试环境中去熟悉调试方法和调试工具,这才会加深理解. gdb能够做什么?它可以按照你的定义,随心 ...
- 【实习记】2014-08-19升级vim配置YouCompleteMe并debug的过程+qtcreator有语言包没法换语言
做了个小项目,有空闲可以做点事了. 偶然查资料看见YouCompleteMe的鼎鼎大名. 演示demo <img src="http://i.imgur.com/0OP4ood ...
- javascript常用对象
A,window对象 window对象是浏览器模型对象的顶层对象 常用属性: screen:客户端的屏幕和显示性能的信息. history:客户端访问过的url信息 location:当前url链接的 ...
- C#让TopMost窗体弹出并置顶层但不获取当前输入焦点的终极办法
为了使程序在弹出窗口时置顶层且不获取系统输入焦点,避免影响用户当前的操作,来电通来电弹屏软件尝试过N多种办法,例如:弹出前保存当前焦点窗口句柄,弹出时因为使用TopMost系统默认将焦点交给了弹出窗口 ...
- Cassandra1.2文档学习(17)—— CQL数据模型(上)
参考文档:http://www.datastax.com/documentation/cql/3.0/webhelp/index.html#cql/ddl/ddl_anatomy_table_c.ht ...
- RHEL 7.2 安装Oracle XE-11.2.0
轻量快捷版本,适合开发 0. /etc/hosts 添加 本机hostname # hostnamepromote.cache-dns.local # cat /etc/hosts127.0.0.1 ...