day36【贪心算法】● 435. 无重叠区间 ● 763.划分字母区间 ● 56. 合并区间

435. 无重叠区间

  • 435. 无重叠区间 | 题目链接

  • 代码随想录 | 讲解链接

  • 题意:给定一个区间的集合 intervals ,其中 intervals[i] = [starti, endi] 。返回 需要移除区间的最小数量,使剩余区间互不重叠 。

      示例 1:
      输入: intervals = [[1,2],[2,3],[3,4],[1,3]]
      输出: 1
      解释: 移除 [1,3] 后,剩下的区间没有重叠。
      
      示例 2:
      输入: intervals = [ [1,2], [1,2], [1,2] ]
      输出: 2
      解释: 你需要移除两个 [1,2] 来使剩下的区间没有重叠。
      
      示例 3:
      输入: intervals = [ [1,2], [2,3] ]
      输出: 0
      解释: 你不需要移除任何区间,因为它们已经是无重叠的了。
    
  • 思路:和day35中的射气球的题很像,把重叠区间的个数求出来就可以了。

class Solution {
    public int eraseOverlapIntervals(int[][] intervals) {
        Arrays.sort(intervals, (a, b)-> Integer.compare(a[0],b[0]));

        //记录重叠区间的数量
        int count = 0;
        for(int i = 1; i < intervals.length; i++) {
            //如果有重叠的区间,count自加。
            //更新i的右边界,在i的右边界之前,和i-1都是重合的部分了。
            if(intervals[i][0] < intervals[i-1][1]) {
                count++;
                intervals[i][1] = Math.min(intervals[i-1][1], intervals[i][1]);
            }
        }
        return count;
    }
}

763.划分字母区间

  • 763.划分字母区间 | 题目链接

  • 代码随想录 | 讲解链接

  • 题意:给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段,同一字母最多出现在一个片段中。

    划分结果需要满足:将所有划分结果按顺序连接,得到的字符串仍然是 s 。

    返回一个表示每个字符串片段的长度的列表。

      示例 1:
      输入:s = "ababcbacadefegdehijhklij"
      输出:[9,7,8]
      解释:
      划分结果为 "ababcbaca"、"defegde"、"hijhklij" 。
      每个字母最多出现在一个片段中。
      像 "ababcbacadefegde", "hijhklij" 这样的划分是错误的,因为划分的片段数较少。
       
      示例 2:
      输入:s = "eccbbbbdec"
      输出:[10]
    
  • 思路:
    统计每一个字符最后出现的位置

    从头遍历字符,并更新字符的最远出现下标,如果找到字符最远出现位置下标和当前下标相等了,则找到了分割点

在这里插入图片描述

class Solution {
    public List<Integer> partitionLabels(String s) {
        //记录每个字母出现的最远位置
        int[] nums = new int[26];
        //每截一段的开始和终止位置
        int start = 0;
        int end = 0;
        //存放每段片段的长度,返回结果
        List<Integer> res = new LinkedList<>();

        //遍历字符串s,统计每个字母出现过的最远距离
        //通过i实时更新最远距离
        for(int i = 0; i < s.length(); i++) {
            nums[s.charAt(i) - 'a'] = i;
        }

        //遍历s,及时更新end,取最大值。直到遍历到i和end相等时,说明此前的字母都出现在此片段中了
        //可以把这个结果长度加入res中,然后更新start为i的后一个,再找下一个片段
        for(int i = start; i < s.length(); i++) {
            end = Math.max(end, nums[s.charAt(i) - 'a']);
            if(i == end) {
                res.add(end - start + 1);
                start = i + 1;
            }
        }
        return res;
    }
}

56. 合并区间

  • 56. 合并区间 | 题目链接

  • 代码随想录 | 讲解链接

  • 题意:以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] = [starti, endi] 。请你合并所有重叠的区间,并返回 一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间 。

      示例 1:
      输入:intervals = [[1,3],[2,6],[8,10],[15,18]]
      输出:[[1,6],[8,10],[15,18]]
      解释:区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].
      
      示例 2:
      输入:intervals = [[1,4],[4,5]]
      输出:[[1,5]]
      解释:区间 [1,4] 和 [4,5] 可被视为重叠区间。
    
  • 思路:按左边界排序后,把一个区间加入结果集中,不断把新的区间和结果集中的边界做对比。不断更新右边界即可。

class Solution {
    public int[][] merge(int[][] intervals) {
        Arrays.sort(intervals, (a,b) -> Integer.compare(a[0],b[0]));

        //结果列表
        LinkedList<int[]> res = new LinkedList<>();
        //先把第一个区间放进去
        res.add(intervals[0]);
        //遍历时,不断拿出结果集中最后一个区间与遍历的区间做比较
        for(int i = 1; i < intervals.length; i++) {
            //如果遍历的左边界比结果集中最后一个区间的右边界小,那就可以更新结果集中的右边界了
            //左边界不用更新,因为一定是之前的更小,因为一开始已经按左边界的大小排序了
            if(intervals[i][0] <= res.getLast()[1]) {
                int start = res.getLast()[0];
                int end = Math.max(res.getLast()[1], intervals[i][1]);
                res.removeLast();
                res.add(new int[]{start, end});
            } else { //如果不重合,就直接把新的加进来
                res.add(new int[]{intervals[i][0],intervals[i][1]});
            }
        }
        return res.toArray(new int[res.size()][]);
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

xuwuuu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值