POJ-1328-放置雷达
这是一道贪心的题目,首先我们要知道,我们放置雷达的话我们可以做一个转换,就是已知岛屿的点坐标的时候,我们可以算一下,这个点以d为半径与x轴交点之间的线段在x轴上的投影,然后我们只需要在这个投影范围内设置一个雷达就可以了。
然后我们读入数据的时候,因为题目中说道,如果计算不出结果我们就输出-1,一般来是可以得到结果的,所以如果出现异常,那肯定是数据的锅。
我们读的时候判断一下y值,如果大于d的值的话,我们就给一个flag标记,然后在读入完成之后,直接输出-1,跳出这重循环,继续对下一组数据进行操作。
根据之前的投影,我们只需要记录每个岛屿的投影开始的x值和结束的x值就可以了。
接下来我们就开始计算这雷达的安放数目了,我们可以知道如果在同一个雷达的作用范围可以覆盖多个岛屿的话,我们就在这些处于同一范围的岛屿的最后一个岛的投影开始x值处,安放一个雷达就可以了。
这是因为如果这些岛屿若不能被这个雷达覆盖的话,就是两种情况:
第一种情况就是,其中的一个岛屿的它的左端点已经超过了最右边的那个端点的开始x值,这样就与我们的假设矛盾了,所以不成立。
第二种情况就是,其中的一个岛屿的右端点小于了最右边的那个岛屿的左值,这样子的话,它又与我们的假设相矛盾了。因为如果小于左值,就说明它的范围已经不跟后面的岛是同一个范围的了,一个雷达是不可能覆盖这些岛屿的。
我们执行的时候,也是进行两重循环,第一重循环推进岛屿,第二重循环就从上次未覆盖的岛屿开始直到一重循环的i值-1的位置,进行比对,比较这之间的岛屿的结束位置是否大于岛屿i的开始位置,如果大于,就不做任何操作。
如果小于就说明除了岛屿i之外,上次未覆盖岛屿到i-1的岛屿是处在同一个雷达范围之下的,我们在i-1直接设置一个雷达即可。
这样子的话,就是ans++的操作,然后更新最新的未被覆盖的岛屿的位置,然后跳出内层循环。
但是ans这个值的话,我们要给它设置成1,因为如果只有一个岛屿的话,我们进入循环之后就会发现i=0,此时第一次进入,最新未被覆盖的岛屿位置也是0,然后我们无法进入内层循环,然后就无法进行++操作。
但是我们设置为1的目的并不是为了第一个岛,这也与我们的初衷相悖,与我们的思路不一样,我们设置的目的是为了给最后一个岛进行加一的操作。
我们可以分析一下, 最后一个岛的情况,第一,它被之前的岛覆盖了,外层循环走到最后一个岛,内层循环进行比较,然后发现没有结束的岛屿位置,也就是下一次的岛屿不能全部覆盖的那个岛屿位置。
我们失去了结束的位置,然后ans就不能++,所以我们直接加1。另一种情况就是最后一个岛屿也不被覆盖,但是内层循环依旧找不到结束的位置,这样内层循环结束,外层循环结束,ans依旧没有加加。
所以我们直接给它设置成1,至于贪心算法的正确性其实还有几句话,但是看不看都已经无所谓了,直接看代码去吧。
正确性:
假设你觉得有一个雷达的放置序列是最少的,最佳的,然后我们按照上面的方法得出的是另一个最佳结果。
好,我们来比较这两个序列,设它们的开始是x1和y1(都是x的坐标)。
如果x1<y1,此时,x1所覆盖的岛屿必能被y1所覆盖,这时候我们就可以用y1替换x1。
如果x1>y1,这时候又分为两种情况,第一就是y1<x1<y2,因为x1不能覆盖y2范围内的岛屿,所以我们这时候放置的y1不会有损失,我们可以覆盖所有的岛屿。
如果它大于x2,这时候,根据我们的贪心策略,如果我们在i放置雷达,是因为我们如果在i+1的位置放置雷达,那就无法覆盖i位置上的雷达了。
所以如果x1>y2的话,它本身也无法覆盖1位置上的雷达,所以它本身就不是一个正确的策略了。
所以贪心解法就是正确的。
#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
const double eps=1e-6;
struct Island {
double Sx,Ex;
bool operator < (const Island &b)const {
return Ex<b.Ex;
}
}island[1010];
int main()
{
int n,d,x,y,cnt=1;
while (~scanf("%d%d",&n,&d)&&n&&d) {
int error=0;
for (int i=0;i<n;i++) {
scanf("%d%d",&x,&y);
if (y>d)
error=1;
island[i].Sx=x-sqrt(d*d-y*y);
island[i].Ex=x+sqrt(d*d-y*y);
}
if (error) {
printf("Case %d: %d\n",cnt++,-1);
continue;
}
sort(island,island+n);
int end=0;
int ans=1;
for (int i=0;i<n;i++) {
for (int j=end;j<i;j++) {
if (island[i].Sx-island[j].Ex<eps) {
continue;
}
else {
ans++;
end=i;
break;
//此时i-1已经被覆盖
}
}
}
printf("Case %d: %d\n",cnt++,ans);
}
return 0;
}
POJ-1328-放置雷达的更多相关文章
- poj 1328 安雷达问题 贪心算法
题意:雷达如何放置?在xoy二维平面坐标系里面,x轴上方的为岛屿,x轴下方的是雷达要放到位置,如何放使得雷达放的最少? 思路 肯定放在x轴上减少浪费是最好的选择 什么情况下,雷达无法到达呢?--以这个 ...
- POJ 1328 安装雷达 (贪心)
<题目链接> 题目大意: 以x轴为分界,y>0部分为海,y<0部分为陆地,给出一些岛屿坐标(在海中),再给出雷达可达到范围,雷达只可以安在陆地上,问最少多少雷达可以覆盖所以岛屿 ...
- poj 1328
http://poj.org/problem?id=1328 题意:题目大概意思就是有一群孤岛,想要用雷达来监视这些岛屿,但雷达的范围是有限的,所以需要多个雷达,题目就是要你解决最少需要几个雷达,注意 ...
- poj 1328 Radar Installation(贪心)
题目:http://poj.org/problem?id=1328 题意:建立一个平面坐标,x轴上方是海洋,x轴下方是陆地.在海上有n个小岛,每个小岛看做一个点.然后在x轴上有雷达,雷达能覆盖的范 ...
- poj 1328 Radar Installatio【贪心】
题目地址:http://poj.org/problem?id=1328 Sample Input 3 2 1 2 -3 1 2 1 1 2 0 2 0 0 Sample Output Case 1: ...
- 贪心 POJ 1328 Radar Installation
题目地址:http://poj.org/problem?id=1328 /* 贪心 (转载)题意:有一条海岸线,在海岸线上方是大海,海中有一些岛屿, 这些岛的位置已知,海岸线上有雷达,雷达的覆盖半径知 ...
- POJ 1328 Radar Installation【贪心】
POJ 1328 题意: 将一条海岸线看成X轴,X轴上面是大海,海上有若干岛屿,给出雷达的覆盖半径和岛屿的位置,要求在海岸线上建雷达,在雷达能够覆盖全部岛屿情况下,求雷达的最少使用量. 分析: 贪心法 ...
- POJ 1328 Radar Installation 贪心 A
POJ 1328 Radar Installation https://vjudge.net/problem/POJ-1328 题目: Assume the coasting is an infini ...
- 雷达装置 (POJ 1328/ codevs 2625)题解
[问题描述] 假定海岸线是一条无限延伸的直线,陆地在海岸线的一边,大海在另一侧.海中有许多岛屿,每一个小岛我们可以认为是一个点.现在要在海岸线上安装雷达,雷达的覆盖范围是d,也就是说大海中一个小岛能被 ...
- poj 1328 Radar Installation
题目链接:http://poj.org/problem?id=1328 题意:给出海上有n个小岛的坐标,求发出的信号可以覆盖全部小岛的最少的雷达个数.雷达发射信号是以雷达为圆心,d为半径的圆,雷达都在 ...
随机推荐
- 洛谷P1850 换教室(概率dp)
传送门 我的floyd竟然写错了?今年NOIP怕不是要爆零了? 这就是一个概率dp 我们用$dp[i][j][k]$表示在第$i$个时间段,已经申请了$j$次,$k$表示本次换或不换,然后直接暴力转移 ...
- Spring boot整合redis实现shiro的分布式session共享
我们知道,shiro是通过SessionManager来管理Session的,而对于Session的操作则是通过SessionDao来实现的,默认的情况下,shiro实现了两种SessionDao,分 ...
- Android课程设计第二天界面排版
注意:课程设计只为完成任务,不做细节描述~ 老师叫我们做一个这个样子,然后.. <?xml version="1.0" encoding="utf-8"? ...
- 洛谷 P1199 三国游戏
参考:Solution_ID:17 题解 更新时间: 2016-11-13 21:01 这道题要求最后得到的两方的默契值最大的武将,小涵的默契值大于计算机,首先,我们这个解法获胜的思路是,每个武将对应 ...
- ACM_求第k大元素(两次二分)
求第k大 Time Limit: 6000/3000ms (Java/Others) Problem Description: 给定两个数组A和B,大小为N,M,每次从两个数组各取一个数相乘放入数组C ...
- hashTable 和 hashMap的区别
HashMap是Hashtable的轻量级实现(非线程安全的实现),他们都完成了Map接口,HashMap把Hashtable的contains方法去掉了,改成containsvalue和contai ...
- Error: IO_ERROR : java.io.IOException: Error while connecting Oozie server. No of retries = 5. Exception = Connection refused (Connection refused)解决办法(图文详解)
不多说,直接上干货! 问题详情 解决办法 Copy/Paste oozie.services property tag set from oozie-default.xml to oozie-site ...
- Vue checkbox默认值改变
<label><input v-bind:true-value="1" v-bind:false-value="0" type=" ...
- AJPFX的反射学习笔记
反射是描述 数据结构的结构 属性.方法(数据)元数据 类(数据结构)描述数据的结构-->类也是特殊的对象---->元数据 CLASS类 描述数据结 ...
- SpringBoot 2.x (6):使用Filter、Servlet、Listener
Filter的使用: 1.常见的默认Filter有: characterEncodingFilter hiddenHttpMethodFilter httpPutFormContentFilter r ...