(leetcode算法题)769. 最多能完成排序的块

Q1. 是否能用贪心算法?为什么?

先预设一个策略,每当当前的nums[i]满足可以 "成块",就直接让这个数成块,也就是说之后的遍历过程中不会将这个数在考虑到自己的块内,

"成块" 是指只要只需要将nums[i]放到前面的某个子数组的尾部,然后将这个子数组进行排序,就能得到一个拥有连续自然数的子数组,就称为成块

能够使用谈心算法是因为有如下规律

规律1. 以nums[i]为结尾的成块的子数组,其中的最大值不能小于 i

反证法:假设nums[i]为结尾的成块的子数组,其中最大值小于 i

那么对这个子数组进行排序后,最后一个值即为maxval,且其下标标定位i

子数组最开始的那个下标设为j, 那么子数组中应该有 i - j + 1个元素

又根据成块的定义,这里将会缺少自然数填满i - j + 1个位置矛盾

故,想要成块,子数组的最大值不能小于 i

下面以图示的方法进一步说明,假设红线前的0 1 2已经成块了

如果 nums[7] < 7 那么一定不能成块,因为此时只能有 6 5 4 3 2 1 0 能放入这8个黑框中,

规律2. 以nums[i]为结尾的成块的子数组,其中的最大值不能大于 i

证明与上面类似,矛盾之处在于如果最大值大于 i ,则将会多出来一个元素

所以要想成块只能是maxval == i

cpp 复制代码
class Solution {
public:
    int maxChunksToSorted(vector<int>& arr) {
        int n = arr.size();
        int ret = 0;
        int curmax = 0;
        for(int i = 0; i < n; ++i){
            curmax = max(curmax, arr[i]);
            if(curmax == i){
                ret++;
            }
        }
        return ret;
    }
};
相关推荐
神龙斗士24024 分钟前
Java 数组的定义与使用
java·开发语言·数据结构·算法
Y.O.U..29 分钟前
力扣HOT100-跳跃游戏II
算法·leetcode
hn小菜鸡30 分钟前
LeetCode 3132.找出与数组相加的整数 II
算法·leetcode·职场和发展
微笑尅乐34 分钟前
数组模拟加法——力扣66.加一
算法·leetcode·职场和发展
_不会dp不改名_1 小时前
leetcode_146 LRU缓存
算法·leetcode·缓存
帅帅爱数学2 小时前
DeepMimic论文详细解析:基于示例引导的深度强化学习实现物理仿真角色技能
算法·强化学习
Dream it possible!2 小时前
LeetCode 面试经典 150_哈希表_快乐数(45_202_C++_简单)(哈希表;快慢指针)
leetcode·面试·散列表
IT成长日记2 小时前
【LVS入门宝典】LVS调度算法轮询(RR)深度解析:从原理到实战的公平调度之道
算法·lvs·rr·轮询调度算法
NAGNIP3 小时前
一文搞懂量化、剪枝和知识蒸馏都是什么?
算法
点云SLAM3 小时前
GTSAM 中自定义因子(Custom Factor)的详解和实战示例
算法·机器人·slam·后端优化·gtsam·gtsam自定义因子·因子图