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为半径的圆,雷达都在 ...
随机推荐
- bzoj2064: 分裂(状压dp)
Description 背景: 和久必分,分久必和... 题目描述: 中国历史上上分分和和次数非常多..通读中国历史的WJMZBMR表示毫无压力. 同时经常搞OI的他把这个变成了一个数学模型. 假设中 ...
- 基于 CODING 轻松搞定持续集成
点击观看视频教程 带你一步一步搞定 CODING 持续集成 持续集成加速软件交付 持续集成这个概念是由 Grady Booch 在 1991 年首次提出,随后成为了 DevOps 的核心实践之一.持续 ...
- 安装截图软件shutter
直接进入软件中心安装吧 不知道软件中心安装的软件可以apt upgrade 命令更新么 应该是可以的,感觉软件中心就是个图形化的apt和新立得差不多 shutter可以放大缩小截图, ctrl + 鼠 ...
- python之类的相关名词-继承-
继承:父类有的功能,子类继承后也都有 继承是直接把父类方法写入子类的object里 如果定义的类有很多重复的功能,可以把重复的类定义成父类 静态方法:不需要实例化就可以调用,不可以调用类里面的变量和方 ...
- PopupWindow(3)back,home 键无法关闭popupwindow的解决方案
private PopupWindow mPopupWindow; //popup window 一般popuowindow 要都个显示view,本例子中view模拟菜单. private View ...
- SPRING-BOOT系列之Spring4快速入门
上节 : spring boot简介 接着上章节的spring boot简介,我们会发现boot是基于spring的,其中最重要的就是spring容器了.那么本章着重介绍spring容器装配自定义be ...
- KEIL软件仿真死在等待外部晶振起振
这是由于是Debug里面的设置有问题 主要是下面2项设置 Dialog DLL默认是DCM3.DLL Parameter默认是-pCM3 应改为 Dialog DLL默认是DARMSTM.DLL Pa ...
- 移动端UI自动化Appium测试——Android系统下使用uiautomator viewer查找元素
在利用Appium做自动化测试时,最重要的一步就是获取对应的元素值,根据元素来对对象进行对应的操作,如何获得对象元素呢?Appium Server Console其实提供了一个界面对话框&qu ...
- mongoDB内置文档定义
在最近的设计数据库时,犯了一个低级的错误,就是设置内置文档是定义了错误了,导致数据取不出,去找了很多资料都无法解决.最后看了一了一下自己设置的model文件.配置错误,所以导致数据取不出了. 数据库时 ...
- hash 【模板】
hash 功能: hash一般用于快速判断两个或多个字符串是否匹配. 实现 : 想一想,如果比较两个数子的话是很方便的很快,那么我们把整个字符串看成一个大数. 它是base进制的len位数.但 ...