合并 K 个升序链表

cpp 复制代码
class Solution {
public:
    // 主函数:负责启动分治过程
    ListNode* mergeKLists(vector<ListNode*>& lists) {
        // 1. 如果列表为空,直接返回空(防止空数组报错)
        if (lists.empty()) return nullptr;
        // 2. 调用分治函数,范围从 0 到最后一个索引
        return solve(lists, 0, lists.size() - 1);
    }

private:
    // 分治函数:负责把 K 个链表不断二分
    ListNode* solve(vector<ListNode*>& lists, int l, int r) {
        // 3. 递归终止:如果区间里只有一个链表,直接返回它
        if (l == r) return lists[l];
        // 4. 计算中点,>> 1 相当于除以 2(位运算更快)
        int mid = (l + r) >> 1;
        // 5. 递归处理左边,递归处理右边,并把两边的结果通过 merge 函数合并
        return merge(solve(lists, l, mid), solve(lists, mid + 1, r));
    }

    // 核心合并函数:合并两个有序链表(递归版)
    ListNode* merge(ListNode* a, ListNode* b) {
        // 6. 如果 a 空了,返回 b;如果 b 空了,返回 a(谁不空回谁)
        if (!a || !b) return a ? a : b;

        // 7. 如果 a 节点的值比较小
        if (a->val < b->val) {
            // 8. 让 a 的下一个位置去接"剩下的节点合并后的结果"
            a->next = merge(a->next, b);
            // 9. 选了 a 作为当前最小,返回 a
            return a;
        } else {
            // 10. 如果 b 的值较小,同理让 b 的下一个位置去接合并结果
            b->next = merge(a, b->next);
            // 11. 选了 b 作为当前最小,返回 b
            return b;
        }
    }
};

假设我们的 lists 中有 4个小链表

  • L0 : [1, 4]

  • L1 : [2, 6]

  • L2 : [0, 8]

  • L3 : [3, 7]


第一阶段:分治拆解 (solve 函数)

分治像一棵倒置的树,先把问题不断对半劈开。

  1. 第一层solve(0, 3)

    • 计算 mid = 1

    • 拆分为左边 solve(0, 1) 和右边 solve(2, 3)

  2. 第二层(左)solve(0, 1)

    • 计算 mid = 0

    • 拆分为 solve(0, 0)(直接返回 L0 )和 solve(1, 1)(直接返回 L1)。

  3. 第二层(右)solve(2, 3)

    • 计算 mid = 2

    • 拆分为 solve(2, 2)(直接返回 L2 )和 solve(3, 3)(直接返回 L3)。


第二阶段:递归合并 (merge 函数)

现在开始从底层向上"两两决斗"合并。

1. 合并 L0 [1, 4] 和 L1 [2, 6]
  • 比较 1 和 2 :1 小,选 1。1 的后面去接 merge(L0的4, L1的2)

  • 比较 4 和 2 :2 小,选 2。2 的后面去接 merge(L0的4, L1的6)

  • 比较 4 和 6 :4 小,选 4。4 的后面去接 merge(null, L1的6)

  • 收尾:遇到 null,直接接上 6。

  • 结果 (La)[1, 2, 4, 6]

2. 合并 L2 [0, 8] 和 L3 [3, 7]
  • 同理,经过比较:0 < 3,3 < 7,7 < 8。

  • 结果 (Lb)[0, 3, 7, 8]


第三阶段:终极对决 (La 和 Lb 合并)

这是最后一步:merge(La, Lb),即合并 [1, 2, 4, 6][0, 3, 7, 8]

  1. 比较 1 和 00 胜出。结果链表:0 -> ...

  2. 比较 1 和 31 胜出。结果链表:0 -> 1 -> ...

  3. 比较 2 和 32 胜出。结果链表:0 -> 1 -> 2 -> ...

  4. 比较 4 和 33 胜出。结果链表:0 -> 1 -> 2 -> 3 -> ...

  5. 比较 4 和 74 胜出。结果链表:0 -> 1 -> 2 -> 3 -> 4 -> ...

  6. 比较 6 和 76 胜出。结果链表:0 -> 1 -> 2 -> 3 -> 4 -> 6 -> ...

  7. 收尾La 已经空了,把 Lb 剩下的 [7, 8] 直接贴在后面。

最终冠军[0, 1, 2, 3, 4, 6, 7, 8]


总结这个过程的记忆点

  • Solve 是"向下扎根":直到看到单个链表。

  • Merge 是"向上开花":把拿到的两个链表像拉拉链一样合起来。

  • 递归的魅力:你不需要管中间有多少层,你只需要写好"两个怎么合",递归会自动帮你处理剩下的 K-2 个。

相关推荐
墨染天姬1 天前
[AI]OPENAI的PPO算法
人工智能·算法
cici158741 天前
含风光储燃的微电网能量管理系统(PSO优化)
算法
Das11 天前
图像色彩迁移技术算法及基本原理
算法
发疯幼稚鬼1 天前
二叉树的广度优先遍历
c语言·数据结构·算法·宽度优先
谭欣辰1 天前
C++ DFS 与 BFS 剪枝方法详解
c++·算法·剪枝
Via_Neo1 天前
乘积最大问题
数据结构·算法
CN-Dust1 天前
【C++专题】格式化输出与输入
开发语言·c++·算法
自我意识的多元宇宙1 天前
数据结构----插入排序
数据结构·算法·排序算法
im_AMBER1 天前
Leetcode 162 除了自身以外数组的乘积 | 接雨水
开发语言·javascript·数据结构·算法·leetcode
Westward-sun.1 天前
YOLO目标检测算法与mAP评估指标详解(附示例)
算法·yolo·目标检测