算法训练 | 贪心算法Part5 | 435.无重叠区间、763.划分字母区间、56. 合并区间、738.单调递增的数字

06-13 1014阅读

目录

算法训练 | 贪心算法Part5 | 435.无重叠区间、763.划分字母区间、56. 合并区间、738.单调递增的数字
(图片来源网络,侵删)

435.无重叠区间

贪心算法

763.划分字母区间

贪心算法

56. 合并区间

贪心算法

738.单调递增的数字

贪心算法


435.无重叠区间

  • 题目链接:435. 无重叠区间 - 力扣(LeetCode)

  • 文章讲解:代码随想录

    贪心算法
    • 解题思路

      • 按照右边界排序,还是按照左边界排序呢?其实都可以。主要就是为了让区间尽可能的重叠。

      • 按照右边界排序,从左向右记录非交叉区间的个数。最后用区间总数减去非交叉区间的个数就是需要移除的区间个数了。此时问题就是要求非交叉区间的最大个数。

    • 解题步骤

      • 区间,1,2,3,4,5,6都按照右边界排好序。

      • 当确定区间 1 和 区间2 重叠后,取 区间1 和 区间2 右边界的最小值,因为这个最小值之前的部分一定是 区间1 和区间2 的重合部分,如果这个最小值也触达到区间3,那么说明 区间 1,2,3都是重合的。

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

      • 区间4结束之后,再找到区间6,所以一共记录非交叉区间的个数是三个。

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

    • 代码一:左区间优化 ⭐

      class Solution {
      public:
          static bool cmp (const vector& a, const vector& b) {
              return a[0]  
      

      763.划分字母区间

      • 题目链接:763. 划分字母区间 - 力扣(LeetCode)

      • 文章讲解:代码随想录

        贪心算法
        • 解题思路

          • 题目要求同一字母最多出现在一个片段中,那么如何把同一个字母的都圈在同一个区间里。

          • 在遍历的过程中相当于是要找每一个字母的边界,如果找到之前遍历过的所有字母的最远边界,说明这个边界就是分割点了。此时前面出现过所有字母,最远也就到这个边界了。

        • 解题步骤

          • 统计每一个字符最后出现的位置

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

        • 代码一:贪心算法

          class Solution {
          public:
              vector partitionLabels(string S) {
                  int hash[27] = {0}; // i为字符,hash[i]为字符出现的最后位置
                  for (int i = 0; i  
          

          56. 合并区间

          • 题目链接:56. 合并区间 - 力扣(LeetCode)

          • 文章讲解:代码随想录

            贪心算法
            • 解题思路

              • 先排序,让所有的相邻区间尽可能的重叠在一起,按左边界,或者右边界排序都可以,处理逻辑稍有不同。

              • 按照左边界从小到大排序之后,如果 intervals[i][0] strNum[i]的情况,让strNum[i - 1]减一,但此时如果strNum[i - 1]减一了,可能又小于strNum[i - 2]。举个例子,数字:332,从前向后遍历的话,那么就把变成了329,此时2又小于了第一位的3了,真正的结果应该是299。

              • 那么从后向前遍历,就可以重复利用上次比较得出的结果了,从后向前遍历332的数值变化为:332 -> 329 -> 299确定了遍历顺序之后,那么此时局部最优就可以推出全局

            • 代码一:贪心算法

              // 时间复杂度:O(n),n 为数字长度
              // 空间复杂度:O(n),需要一个字符串,转化为字符串操作更方便
              class Solution {
              public:
                  int monotoneIncreasingDigits(int N) {
                      string strNum = to_string(N);
                      // flag用来标记赋值9从哪里开始
                      // 设置为这个默认值,为了防止第二个for循环在flag没有被赋值的情况下执行
                      int flag = strNum.size();
                      for (int i = strNum.size() - 1; i > 0; i--) {
                          if (strNum[i - 1] > strNum[i] ) {
                              flag = i;
                              strNum[i - 1]--;
                          }
                      }
                      for (int i = flag; i  
              
VPS购买请点击我

文章版权声明:除非注明,否则均为主机测评原创文章,转载或复制请以超链接形式并注明出处。

目录[+]