链表篇(一)——合并两个有序链表

目录

​编辑

一、题目本质

二、解法

1.迭代

2.递归

三、复杂度分析

四、变形训练

变体1:去重合并

变体2:降序合并

变体3:K路归并


一、题目本质

线性归并(Merge)------归并排序核心操作

二、解法

1.迭代

cpp 复制代码
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {
    ListNode dummy(0);          // 栈上哨兵,省去delete
    ListNode* tail = &dummy;    // tail 始终指向结果链表末尾
    
    while (list1 && list2) {
        if (list1->val <= list2->val) {
            tail->next = list1;
            list1 = list1->next;
        } else {
            tail->next = list2;
            list2 = list2->next;
        }
        tail = tail->next;
    }
    tail->next = list1 ? list1 : list2;  // 拼接剩余
    return dummy.next;
}

2.递归

cpp 复制代码
ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) {
    if (!l1) return l2;
    if (!l2) return l1;
    if (l1->val <= l2->val) {
        l1->next = mergeTwoLists(l1->next, l2);
        return l1;
    } else {
        l2->next = mergeTwoLists(l1, l2->next);
        return l2;
    }
}

三、复杂度分析

四、变形训练

变体1:去重合并

要求:相同值的结点只保留一个,两个指针都前进。

变体2:降序合并

(1)当输入也为降序时直接修改比较符号。

(2)输入为升序,输出为降序采用头插法

cpp 复制代码
ListNode* mergeTwoListsDesc(ListNode* l1, ListNode* l2) {
    ListNode* result = nullptr;  // 无哨兵,直接头插
    while (l1 && l2) {
        ListNode** smaller = (l1->val <= l2->val) ? &l1 : &l2;
        ListNode* node = *smaller;
        *smaller = (*smaller)->next;
        node->next = result;  // 头插到结果
        result = node;
    }
    // 剩余部分也要头插(略),或先合并再反转整个链表更简单
    return result;
}

变体3:K路归并

(1)利用分支归并,两两合并。

cpp 复制代码
ListNode* mergeKListsDivide(vector<ListNode*>& lists, int l, int r) {
    if (l == r) return lists[l];
    if (l > r) return nullptr;
    int mid = (l + r) / 2;
    ListNode* left = mergeKListsDivide(lists, l, mid);
    ListNode* right = mergeKListsDivide(lists, mid + 1, r);
    return mergeTwoLists(left, right);  // 复用两链表合并
}

(2)采用最小堆,优先队列

cpp 复制代码
ListNode* mergeKLists(vector<ListNode*>& lists) {
    // 小顶堆:按结点值排序
    auto cmp = [](ListNode* a, ListNode* b) { return a->val > b->val; };
    priority_queue<ListNode*, vector<ListNode*>, decltype(cmp)> pq(cmp);
    
    // 每个链表的头结点入堆
    for (auto head : lists) {
        if (head) pq.push(head);
    }
    
    ListNode dummy(0);
    ListNode* tail = &dummy;
    
    while (!pq.empty()) {
        ListNode* node = pq.top(); pq.pop();
        tail->next = node;
        tail = tail->next;
        if (node->next) pq.push(node->next);  // 下一个结点入堆
    }
    return dummy.next;
}
相关推荐
ZzYH22几秒前
文献阅读 260529-Burning Questions: Research Data, Tools, and Insights
笔记
逻辑君3 分钟前
Foresight研究报告【20260010】
人工智能·算法·机器学习
一切皆是因缘际会5 分钟前
AI高速迭代下的技术风险与理性突围
大数据·数据结构·人工智能·架构
玖玥拾5 分钟前
C/C++ 基础笔记(一)
c语言·c++·笔记
weixin_468466857 分钟前
大语言模型原理新手入门指南
人工智能·python·算法·语言模型·自然语言处理·transformer·注意力机制
z2005093011 分钟前
今日算法(回溯找IP,加检测)
算法·leetcode
sheeta199815 分钟前
LeetCode 补拙笔记 日期:2026.05.29 题目:1559. 二维网格图中探测环
笔记·算法·leetcode
罗超驿17 分钟前
10.滑动窗口解决:无重复字符的最长子串 | LeetCode 3 Java 题解
java·算法·leetcode·面试
罗超驿17 分钟前
8.【LeetCode 18】四数之和 —— Java 排序 + 双指针解法详解
算法·leetcode·职场和发展
菜菜的顾清寒21 分钟前
HOT100力扣(40) 动态规划-爬楼梯
算法·leetcode·动态规划