leetcode 768. 最多能完成排序的块 II

题目

https://leetcode.cn/problems/max-chunks-to-make-sorted-ii/description/

单调栈解法复现:输入 [1,1,0,0,1]

让我们一步步复现单调栈解法在输入 [1,1,0,0,1] 时的执行过程:

1. 处理第一个元素 1

  • 栈操作 :栈为空,直接压入 1
  • 栈状态[1]
  • 块划分 :当前块为 [1],最大值为 1

2. 处理第二个元素 1

  • 比较1 == 栈顶(1),满足 num >= st.top()
  • 栈操作 :直接压入 1
  • 栈状态[1, 1]
  • 块划分 :新增块 [1],最大值为 1
  • 当前块序列[1], [1]

3. 处理第三个元素 0

  • 比较0 < 栈顶(1),触发合并逻辑:
    1. 弹出栈顶 1,记录 maxVal = 1
    2. 栈非空,且 st.top() = 1 > 0,继续弹出。
    3. 弹出 1 后,栈空,退出循环。
    4. 压入 maxVal = 1
  • 栈状态[1]
  • 块划分 :合并块 [1, 1, 0],最大值为 1
  • 当前块序列[1, 1, 0]

4. 处理第四个元素 0

  • 比较0 < 栈顶(1),触发合并逻辑:
    1. 弹出栈顶 1,记录 maxVal = 1
    2. 栈空,退出循环。
    3. 压入 maxVal = 1
  • 栈状态[1]
  • 块划分 :合并块 [1, 1, 0, 0],最大值为 1
  • 当前块序列[1, 1, 0, 0]

5. 处理第五个元素 1

  • 比较1 == 栈顶(1),满足 num >= st.top()
  • 栈操作 :直接压入 1
  • 栈状态[1, 1]
  • 块划分 :新增块 [1],最大值为 1
  • 最终块序列[1, 1, 0, 0], [1]

最终结果

  • 栈大小2
  • 最优划分[1, 1, 0, 0], [1]
  • 块数2

关键分析

  1. 合并逻辑的触发

    当遇到 0 时,由于 0 < 1,需要将前面的块合并,确保块的最大值单调递增。否则,若保留 [1], [1], [0],排序后 0 会出现在 1 前面,导致错误。

  2. 保留最大值的意义

    合并后压入 maxVal = 1,确保后续块的最大值必须 >=1,从而保证排序后整体有序。例如,若第五个元素是 2,则可形成新块 [2],最终块序列为 [1, 1, 0, 0], [2]

  3. 验证排序结果

    • 每个块内部排序:[0, 0, 1, 1], [1]
    • 连接后:[0, 0, 1, 1, 1],与原数组排序结果一致。

代码复现

cpp 复制代码
#include <iostream>
#include <vector>
#include <stack>
using namespace std;

int maxChunksToSorted(vector<int>& arr) {
    stack<int> st;
    
    for (int num : arr) {
        cout << "处理元素: " << num << ", 栈状态: ";
        // 打印当前栈
        vector<int> temp;
        stack<int> copy = st;
        while (!copy.empty()) {
            temp.push_back(copy.top());
            copy.pop();
        }
        for (int i = temp.size() - 1; i >= 0; i--) {
            cout << temp[i] << " ";
        }
        cout << endl;
        
        if (st.empty() || num >= st.top()) {
            st.push(num);
            cout << "  压入 " << num << endl;
        } else {
            int maxVal = st.top();
            st.pop();
            cout << "  弹出 " << maxVal << ", 记录 maxVal = " << maxVal << endl;
            
            while (!st.empty() && st.top() > num) {
                cout << "  继续弹出 " << st.top() << endl;
                st.pop();
            }
            
            st.push(maxVal);
            cout << "  压入 maxVal = " << maxVal << endl;
        }
    }
    
    return st.size();
}

int main() {
    vector<int> arr = {1, 1, 0, 0, 1};
    cout << "最终块数: " << maxChunksToSorted(arr) << endl;
    return 0;
}

运行上述代码可观察到每一步的栈变化,验证整个处理过程。

相关推荐
我爱cope19 分钟前
【Agent智能体4 | 智能体AI的应用】
数据库·人工智能·职场和发展
全糖可乐气泡水1 小时前
Codex适配国产信创环境安装部署与技术适配全解析
开发语言·git·python·算法·百度
h_a_o777oah1 小时前
状态机+划分型 DP :深度解析K-划分问题下 DP 状态的转移逻辑(洛谷P2679 P2331 附C++代码)
c++·算法·动态规划·acm·状态机dp·划分型dp·滚动数组优化
05候补工程师1 小时前
从算法理想向工程现实的跨越:SLAM 核心架构、思维误区与 Nav2 实战避坑指南
人工智能·算法·安全·架构·机器人
手写码匠2 小时前
Android 17 适配实战指南:新特性解读、隐私变更与迁移全攻略
人工智能·深度学习·算法·aigc
珊瑚里的鱼3 小时前
leetcode42雨水
算法·leetcode
水木流年追梦3 小时前
大模型入门-大模型的推理策略
开发语言·python·算法·正则表达式·prompt
生成论实验室3 小时前
用事件关系网络重新理解AI(三):激活函数、微调与元学习
人工智能·学习·算法·语言模型·可信计算技术
Narv工程师3 小时前
嵌入式机器人控制器算力评估:从DMIPS到WCET的完整指南
人工智能·算法·机器学习
蒟蒻的贤3 小时前
实训1227
算法