题目
数轴上有 n (1<=n<=25000)个闭区间 [ai, bi],选择尽量少的区间覆盖一条指定线段 [1, t]( 1<=t<=1,000,000)。
覆盖整点,即(1,2)+(3,4)可以覆盖(1,4)。
不可能办到输出-1。
Input
第一行:N和T。
第二行至N+1行: 每一行一个闭区间。
Output
选择的区间的数目,不可能办到输出-1。
Sample Input
3 10
1 7
3 6
6 10
Sample Output
2
思路
这道题让我深深的感受到了wa的快乐
本题采用贪心算法,在数轴上从左到右对区间进行排序,即区间按b从小到大排序(b相同时a从大到小排序),然后对区间按序考虑是否选择,以覆盖指定区间。
定义一个函数find(int A,int &start),从序区间数组的s[start]开始找到左端点小于A且s[i].b-A最大的区间。若找到则返回s[i].b,否则返回-1。在该函数中需注意max初始化为0而不是A(易错数据5);找到的情况中包含max=A,因为A也需覆盖(易错数据4)。
循环,遍历所有区间,不断进行find(int A,int &start),直到返回值为-1无法覆盖或找到解。
Hint
这道题输入数据很多,请用scanf而不是cin。
代码
#include <stdio.h>
#include <algorithm>
using namespace std;
struct section{
int a,b;
bool operator <(const section& s)const{
if(a!=s.a)
return a<s.a;
else
return b>s.b;
}
}s[25000];
int n,t;
int find(int A,int& start){
int i,max=0;//max初始化为0,不要初始化为A
for(i=start;i<n&&s[i].a<=A;i++){
if(s[i].b>max){
max=s[i].b;
}
}
start=i;
if(max<A){//找不到。A也是要覆盖的值,故若区间的b=A则可选,故为<
return -1;
}
return max;
}
int main() {
scanf("%d%d",&n,&t);
for(int i=0;i<n;i++){
scanf("%d%d",&s[i].a,&s[i].b);
}
sort(s,s+n);
int A=1,start=0,count=0,flag=0;;
while(true){
flag=find(A,start);
count++;
if(flag==-1){
printf("-1\n");
break;
}
if(flag>=t){
printf("%d\n",count);
break;
}
A=flag+1;
}
return 0;
}
易错数据
Input 1
4 100
21 50
50 81
1 20
80 99
Output 1
-1
Input 2
3 10
1 3
4 6
7 10
Output 2
3
Input 3
3 10
2 4
3 7
6 10
Output 3
-1
Input 4
6 10
1 5
2 6
3 4
4 5
7 10
2 3
Output 4
3
Input 5
1 3
1 2
Output 5
-1
Input 6
3 6
2 3
4 5
6 6
Output 6
-1
Input 7
2 4
1 2
3 4
Output 7
2