程序设计思维 week3 作业C-区间覆盖

题目

数轴上有 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

易错数据来自:参考博客1参考博客2,🐎一下。
题目链接

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值