题目链接:
https://vjudge.net/problem/POJ-1328
题目大意:
假设陆地的海岸线是一条无限延长的直线,海岛是一个个的点,现需要在海岸线上安装雷达,使整个雷达系统能够覆盖到所有的海岛。雷达所能覆盖的区域是以雷达为圆心半径为d的圆,我们用指标坐标系来描述,海岸线就是x轴,现在给出每个海岛的坐标与雷达的半径d,请编写一个程序计算出最少需要多少个雷达才能够将所有海岛全部覆盖?
思路:
该题题意是为了求出能够覆盖所有岛屿的最小雷达数目,每个小岛对应x轴上的一个区间,在这个区间内的任何一个点放置雷达,则可以覆盖该小岛,区间范围的计
算用[x-sqrt(d*d-y*y),x+sqrt(d*d-y*y)];这样,问题即转化为已知一定数量的区间,求最小数量的点,使得每个区间内斗至少存在一个点。每次迭代对于第一个区间,选
择最右边一个点, 因为它可以让较多区间得到满足, 如果不选择第一个区间最右一个点(选择前面的点), 那么把它换成最右的点之后,以前得到满足的区间, 现在仍然
得到满足, 所以第一个区间的最右一个点为贪婪选择, 选择该点之后, 将得到满足的区间删掉,进行下一步迭代, 直到结束。
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
struct node
{
double x, y;
bool operator < (const node a)const
{
return y < a.y || y == a.y && x > a.x;
}
};
node a[];
int main()
{
int n, d, Case = ;
while(scanf("%d%d", &n, &d) != EOF)
{
if(!n && !d)break;
int x, y;
double r;
bool flag = ;
for(int i = ; i < n; i++)
{
scanf("%d%d", &x, &y);
if(y > d)
{
flag = ;
continue;
}
r = sqrt(1.0 * d * d - y * y);
a[i].x = x - r;
a[i].y = x + r;
//printf("%lf %lf", a[i].x, a[i].y);
}
printf("Case %d: ", ++Case);
if(flag)cout<<"-1"<<endl;
else
{
sort(a, a + n);
int ans = ;
for(int i = ; i < n; i++)
{
double now = a[i].y;
ans++;
for(; i < n; i++)
{
if(a[i].x > now)break;
}
if(i == n)break;
i--;
}
cout<<ans<<endl;
}
}
return ;
}