无重叠区间

在这里插入图片描述
看到这道题目都冥冥之中感觉要排序,但是究竟是按照右边界排序,还是按照左边界排序呢?」

这其实是一个难点!

按照右边界排序,就要从左向右遍历,因为右边界越小越好,只要右边界越小,留给下一个区间的空间就越大,所以从左向右遍历,优先选右边界小的。

按照左边界排序,就要从右向左遍历,因为左边界数值越大越好(越靠右),这样就给前一个区间的空间就越大,所以可以从右向左遍历。

如果按照左边界排序,还从左向右遍历的话,要处理各个区间右边界的各种情况。

「我先来按照右边界排序,从左向右记录非交叉区间的个数。最后用区间总数减去非交叉区间的个数就是需要移除的区间个数了」。

此时问题就是要求非交叉区间的最大个数。

右边界排序之后,局部最优:优先选右边界小的区间,所以从左向右遍历,留给下一个区间的空间大一些,从而尽量避免交叉。全局最优:选取最多的非交叉区间。

局部最优推出全局最优,试试贪心!

这里记录非交叉区间的个数还是有技巧的,如图:
在这里插入图片描述
区间1,2,3,4,5都按照右边界排好序。

每次取非交叉区间的时候,都是选右边界最小的来做分割点(这样留给下一个区间的空间就越大),所以第一条分割线就是区间1结束的位置。

接下来就是找大于区间1结束位置的区间,是从区间4开始。「为什么不从区间5开始?别忘已经是按照右边界排序的了」。

区间4结束,所以一共记录非交叉区间的个数是二个。

总共区间个数为5,减去非交叉区间的个数2。移除区间的最小数量就是3。

按照右边界排序

时间复杂度:O(nlogn) ,有一个快排
空间复杂度:O(1)
class Solution {
public:
    // 按照区间右边界排序
    static bool cmp(vector<int>a,vector<int> b){
        return a[1]<b[1];
    }
    int eraseOverlapIntervals(vector<vector<int>>& intervals) {
        sort(intervals.begin(),intervals.end(),cmp);

        int res=1;
        int minRight=intervals[0][1];
        for(int ii=1;ii<intervals.size();ii++){
            if(intervals[ii][0]>=minRight) {// 记录非交叉区间的个数
             res++;
             minRight=intervals[ii][1];// 记录区间分割点
            }
        }
        return intervals.size()-res;
    }
};

按照左边界排序1

class Solution {
public:
    static bool cmp(vector<int>a,vector<int> b){
        if(a[0]==b[0]) return a[1]<b[1];
        return a[0]<b[0];
    }
    int eraseOverlapIntervals(vector<vector<int>>& intervals) {
        sort(intervals.begin(),intervals.end(),cmp);

        int res=0;
        for(int ii=1;ii<intervals.size();ii++){
            if(intervals[ii][0]<intervals[ii-1][1]) {
             res++;
             //!intervals[ii][1]=intervals[ii-1][1];  【erro】
             intervals[ii][1]=min(intervals[ii-1][1],intervals[ii][1]);
            }
        }
        return res;
    }
};

按照左边界排序2

class Solution {
public:
    // 按照区间左边界从大到小排序
    static bool cmp (const vector<int>& a, const vector<int>& b) {
        return a[0] < b[0];
    }
    int eraseOverlapIntervals(vector<vector<int>>& intervals) {
        if (intervals.size() == 0) return 0;
        sort(intervals.begin(), intervals.end(), cmp);

        int result = 1;
        for (int i = 1; i < intervals.size(); i++) {
            if (intervals[i][0] >= intervals[i - 1][1]) { // 需要要把> 改成 >= 就可以了
                result++; // 需要一支箭
            }
            else {  
                intervals[i][1] = min(intervals[i - 1][1], intervals[i][1]); // 更新重叠气球最小右边界
            }
        }
        return intervals.size() - result;
    }
};

总结

难点一:一看题就有感觉需要排序,但究竟怎么排序,按左边界排还是右边界排。
难点二:排完序之后如何遍历,如果没有分析好遍历顺序,那么排序就没有意义了。
难点三:直接求重复的区间是复杂的,转而求最大非重复区间个数。
难点四:求最大非重复区间个数时,需要一个分割点来做标记。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值