【贪心】区间问题:选点、分组、覆盖以及最大不相交数量

文章介绍了贪心问题中的区间问题,包括区间选点、最大不相交区间数量、区间分组和区间覆盖。对于这些问题,通常策略是先按端点排序,然后通过贪心策略选择最优解,例如选择右端点或左端点来减少交集。文章提供了具体的算法实现和思路解析。
摘要由CSDN通过智能技术生成

贪心问题是一种很复杂的问题,因为没有一个固定的做法,基本就是每个问题当场思考。贪心问题可以总结成一句话:要得到整体最优解,先从局部最优解开始

本篇文章讲解贪心问题中的一类:区间问题,因为这类问题使用频率较大且做法较为统一。

1、区间选点

原题 acwing 区间选点

给定 N 个闭区间 [ a i a_i ai, b i b_i bi],请你在数轴上选择尽量少的点,使得每个区间内至少包含一个选出的点。
输出选择的点的最小数量。
位于区间端点上的点也算作区间内。
输入格式
第一行包含整数 N,表示区间数。
接下来 N 行,每行包含两个整数 a i a_i ai, b i b_i bi,表示一个区间的两个端点。
输出格式
输出一个整数,表示所需的点的最小数量。
数据范围
1≤N≤105,
−109 a i a_i ai b i b_i bi≤109
输入样例

3
-1 1
2 4
3 5

输出样例

2

做法就是:

  1. 先把所有区间按右端点从小到大排序;
  2. 依次枚举每个区间,如果已经包含一个区间内的点,那么就pass;否则就把点选为右端点
  3. 最后选出的点数就是答案。

道理就是这样,凭个人感觉差不多能感觉这样做是对的。因为每次选的都是右端点,也即尽可能地往其他区间上凑,每次都刻意地去贴近其他区间,到最后就能得到正确答案。但是这样说是不严谨的,下面给出一个严谨的证明。
在这里插入图片描述
那么只要按照上面的流程用代码实现即可:

#include <iostream>
#include <algorithm>

using namespace std;

const int N = 100010;

struct Range{
    int l, r;
    
    //重载小于号,以右端点排序
    bool operator< (const Range &W)const
    {
        return r < W.r;
    }
}range[N];

int n;

int main()
{
    cin >> n;
    
    for (int i = 0; i < n; i ++ )
    {
        int l, r;
        scanf("%d%d", &l, &r);
        range[i] = {l, r};
    }
    
    //以右端点排序
    sort(range, range + n);
    
    //res记录答案,ed记录当前选点
    int res = 0, ed = -2e9;
    for (int i = 0; i < n; i ++ )
    	//如果区间左端点都大于当前选点,说明选点不在当前区间
        if (range[i].l > ed)
        {
        	//那么就再选一个新的点
            res ++ ;
            ed = range[i].r;
        }
        
    cout << res << endl;
    
    return 0;
}

2、最大不相交区间数量

原题acwing 最大不相交区间数量

给定 N 个闭区间 [ a i a_i ai, b i b_i bi],请你在数轴上选择若干区间,使得选中的区间之间互不相交(包括端点)。
输出可选取区间的最大数量。
输入格式
第一行包含整数 N,表示区间数。
接下来 N 行,每行包含两个整数 a i a_i ai, b i b_i bi,表示一个区间的两个端点。
输出格式
输出一个整数,表示可选取区间的最大数量。
数据范围
1≤N≤105,
−109 a i a_i ai b i b_i bi≤109
输入样例
3
-1 1
2 4
3 5
输出样例
2

这道题目和区间选点完全一样。至于为什么一样呢,这里只给出字面上的解释:因为有公共点就说明这几个区间是相交的,那么要尽可能的减少相交的区间,就尽可能的减少公共点。所以最少有几个公共点就说明最多有几个互不相交的区间。具体证明较为复杂,不做详细说明。
本题题解和上一题完全一样,复制粘贴即可通过。

3、区间分组

原题 acwing 区间分组

给定 N 个闭区间 [ a i a_i ai, b i b_i bi],请你将这些区间分成若干组,使得每组内部的区间两两之间(包括端点)没有交集,并使得组数尽可能小。
输出最小组数。
输入格式
第一行包含整数 N,表示区间数。
接下来 N 行,每行包含两个整数 a i a_i ai, b i b_i bi,表示一个区间的两个端点。
输出格式
输出一个整数,表示最小组数。
数据范围
1≤N≤105,
−109 a i a_i ai b i b_i bi≤109
输入样例
3
-1 1
2 4
3 5
输出样例
2

做法:

  1. 左端点从小到大排序;
  2. 从前往后依次枚举区间,判断每个区间是否能放到某个组中
    判断方法:已有的所有组中一定有一个右端点最小值,判断一下当前区间的左端点是否大于最小值,如果大于,就说明可以放到组中(因为左端点大于某个组的右端点,说明不相交);否则,说明这个区间和当前所有组都有交集(因为最小的右端点都和它有交集了,那其他右端点一定也会有),这时候就要新开一个组了。

下面给出一个证明:
在这里插入图片描述
由于这里要记录所有组右端点的最小值,所以考虑用小根堆来实现。其中堆中存的是所有组的右端点。给出题解:

#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>

using namespace std;

const int N = 100010;

struct Range{
    int l, r;
    
    bool operator< (const Range &W)const
    {
        return l < W.l;
    }
}range[N];

int n;

int main()
{
    cin >> n;
    
    for (int i = 0; i < n; i ++ )
    {
        int l, r;
        scanf("%d%d", &l, &r);
        range[i] = {l, r};
    }
    
    sort(range, range + n);
    
    priority_queue<int, vector<int>, greater<int>> heap;
    for (int i = 0; i < n; i ++ )
    {
        auto t = range[i];
        if (heap.empty() || heap.top() >= t.l) heap.push(t.r);
        else
        {
            heap.pop();
            heap.push(t.r);
        }
    }
    
    cout << heap.size() << endl;
    
    return 0;
}

4、区间覆盖

原题 acwing 区间覆盖

给定 N 个闭区间 [ a i a_i ai, b i b_i bi],请你将这些区间分成若干组,使得每组内部的区间两两之间(包括端点)没有交集,并使得组数尽可能小。
输出最小组数。
输入格式
第一行包含整数 N,表示区间数。
接下来 N 行,每行包含两个整数 a i a_i ai, b i b_i bi,表示一个区间的两个端点。
输出格式
输出一个整数,表示最小组数。
数据范围
1≤N≤105,
−109 a i a_i ai b i b_i bi≤109
输入样例
3
-1 1
2 4
3 5
输出样例
2

步骤:

  1. 先按左端点排序;
  2. 依次枚举所有区间,选择能覆盖指定区间左端点的区间里,右端点最远的,并且将指定区间左端点更新为此最远的右端点。

字面上来说,就是每次找一个能覆盖的,并且覆盖的最多的,一直这样找下去就能找到最优解。具体证明较为复杂,不过这道题目也比较通俗易懂,所以直接给出题解:

#include <iostream>
#include <algorithm>

using namespace std;

const int N = 100010;

struct Range{
    int l, r;
    
    bool operator< (const Range &W)const
    {
        return l < W.l;
    }
}range[N];

int n;

int main()
{
    int st, ed;
    cin >> st >> ed;
    
    cin >> n;
    
    for (int i = 0; i < n; i ++ )
    {
        int l, r;
        scanf("%d%d", &l, &r);
        range[i] = {l, r};
    }
    
    sort(range, range + n);
    
    int res = 0;
    bool flag = false;
    
    for (int i = 0; i < n; i ++ )
    {
        int j = i, r = -2e9;
        while (j < n && range[j].l <= st)
        {
            r = max(r, range[j].r);
            j ++ ;
        }
        
        //如果所有区间的最大右端点都小于左端点,那就说明无法覆盖
        if (r < st)
        {
            res = -1;
            break;
        }
        
        res ++ ;
        //覆盖到右端点后,即完成
        if (r >= ed)
        {
            flag = true;
            break;
        }
        
        st = r;
        //下一次迭代要从所选区间的下一个开始,这里j其实是所选区间的下一个,并且i会在本次迭代后自增
        //举个例子:所选区间是3,那么此时j是4,因此i是3,但是因为for语句的特性,i会自增,也即4
        i = j - 1;
    }
    
    if (!flag) res = -1;
    cout << res << endl;
    
    return 0;
}

总结

贪心问题一般就是只考虑当前情况的最优方式,累计起来就是整体的最优方式。遇到贪心问题,一般过程为:

  1. 思考每一次怎样做是最优的
  2. 自己设置几个样例检验一下做法的正确性,如果正确就直接写,如果不正确就换策略
  3. 贪心题目一般不考察证明,因为实际写题的时候没有时间证明,只要感觉对,直接试样例即可;
  4. 一定要自己找几组样例,因为题目所给样例可能具有迷惑性

而对于区间问题,一般先按左端点或右端点排序,再思考具体做法。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 如果你想按照某个字段的值的区间分组,可以使用 MySQL 中的 CASE 表达式和 GROUP BY 语句来实现。以下是一个示例: 假设我们有一个表名为 scores,其中包含学生姓名和成绩两列,我们想要按照成绩的区间分组,并统计每个区间内的学生人数。 首先,我们可以使用 CASE 表达式来将成绩分成不同的区间,例如: ``` SELECT CASE WHEN score >= 90 THEN 'A' WHEN score >= 80 THEN 'B' WHEN score >= 70 THEN 'C' WHEN score >= 60 THEN 'D' ELSE 'F' END AS grade, COUNT(*) AS count FROM scores GROUP BY grade; ``` 这个查询会将成绩分成五个区间,并统计每个区间内的学生人数。 注意,我们在 SELECT 语句中使用了 AS 关键字来为 CASE 表达式创建一个别名,这样我们就可以在 GROUP BY 语句中使用这个别名来分组。 另外,COUNT(*) 函数用于计算每个分组中的行数,也就是每个区间内的学生人数。 ### 回答2: MySQL按区间分组可以使用CASE WHEN语句和GROUP BY子句来实现。 假设我们有一个名为"sales"的表,包含了销售数据的信息,其中有一个列"amount"表示销售额。 如果我们希望按照销售额分成三个区间进行分组,可以使用CASE WHEN语句来实现。语法如下: SELECT CASE WHEN amount < 1000 THEN '0-1000' WHEN amount >= 1000 AND amount < 5000 THEN '1000-5000' WHEN amount >= 5000 THEN '5000以上' END AS sales_range, SUM(amount) AS total_amount FROM sales GROUP BY sales_range; 上述查询将会按照销售额区间进行分组,并计算每个区间的总销售额。 例如,如果销售额在0到1000之间,它将被归为'0-1000'区间,如果销售额在1000到5000之间,它将被归为'1000-5000'区间,如果销售额大于等于5000,它将被归为'5000以上'的区间。 最后,我们使用GROUP BY子句将结果按照sales_range进行分组,并计算每个区间的总销售额。 ### 回答3: MySQL 按区间分组,可以使用CASE WHEN语句和GROUP BY子句来实现。 首先,假设有一个包含学生成绩的表格,其中有学生名字和分数两列。我们想按照分数的区间进行分组并计算每个区间内的学生数量。 可以使用CASE WHEN语句来判断每个分数属于哪个区间,然后使用GROUP BY子句按照区间分组。具体步骤如下: ``` SELECT CASE WHEN score >= 90 THEN '90-100' -- 区间判断条件 WHEN score >= 80 THEN '80-89' WHEN score >= 70 THEN '70-79' WHEN score >= 60 THEN '60-69' ELSE '0-59' END AS score_range, -- 将区间结果设为新列 COUNT(*) AS student_count -- 计算每个区间内的学生数量 FROM scores GROUP BY score_range -- 按区间分组 ``` 在上述示例中,首先使用CASE WHEN语句根据分数的不同判断所属的区间(90-100,80-89,70-79,60-69,0-59),然后将区间结果设为新列score_range。 然后,使用GROUP BY子句按照score_range进行分组。最后,使用COUNT(*)统计每个区间内的学生数量。 这样即可实现按区间分组的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值