刷题 - 分治

面试经典 150 题 - 分治

148. 排序链表⭐️⭐️⭐️ - 快慢指针找中间节点 - 归并排序


伪代码:

  • 将链表拆分成两半,返回右半边头节点(左半边头节点就是原始链表头节点)
  • 对左边进行排序并返回左边头节点
  • 对右边进行排序返回右边头节点
  • 传入左右半边头节点,合并两个升序链表
cpp 复制代码
// bug1: 把 cur_a->val <= cur_b->val 写成 cur_a <= cur_b
// 优化1:使用快慢指针找到中间节点,而不是两次遍历

//  分治的思想: 链表排序 = 断开左右半部分连接,对左半部分进行排序 + 对右半部分进行排序 + 合并两个升序链表
//  返回 升序链表的头节点
class Solution {
public:
    // 返回中间节点并拆分链表
    ListNode* splitList(ListNode* head) {
        if (head == nullptr || head->next == nullptr) return nullptr;
        ListNode* slow = head;
        ListNode* fast = head;
        ListNode* prev = nullptr;
        while (fast && fast->next) {
            prev = slow;
            slow = slow->next;
            fast = fast->next->next;
        }
        if (prev) prev->next = nullptr; // 拆分链表
        return slow;
    }
    // 合并两个升序链表 (不使用 dummy_head)
    ListNode* mergeList(ListNode* head_a, ListNode* head_b) {
        if (!head_a) return head_b;
        if (!head_b) return head_a;
        ListNode* merged_head = nullptr;  // 合并后链表的头节点
        ListNode* cur = nullptr;          // 当前操作的节点
        // 初始化 merged_head 和 cur
        if (head_a->val <= head_b->val) {
            merged_head = head_a;
            head_a = head_a->next;
        } else {
            merged_head = head_b;
            head_b = head_b->next;
        }
        cur = merged_head;
        // 合并剩余节点
        while (head_a && head_b) {
            if (head_a->val <= head_b->val) {
                cur->next = head_a;
                head_a = head_a->next;
            } else {
                cur->next = head_b;
                head_b = head_b->next;
            }
            cur = cur->next;
        }
        cur->next = (head_a != nullptr) ? head_a : head_b; 
        return merged_head;
    }

    // 归并排序
    ListNode* sortList(ListNode* head) {
        if (head == nullptr || head->next == nullptr) return head;
        // 拆分链表
        ListNode* mid = splitList(head);
        // 递归排序前半部分和后半部分
        ListNode* left = sortList(head);
        ListNode* right = sortList(mid);
        // 合并排序后的两个链表
        return mergeList(left, right);
    }
};
相关推荐
小欣加油14 小时前
leetcode1926 迷宫中离入口最近的出口
数据结构·c++·算法·leetcode·职场和发展
星恒随风15 小时前
C++ 类和对象入门(五):初始化列表、explicit 和 static 成员详解
开发语言·c++·笔记·学习·状态模式
浪客灿心15 小时前
项目篇:模块设计与实现
数据库·c++
牛油果子哥q15 小时前
【C++ STL vector】C++ STL vector 终极精讲:动态数组底层原理、两倍扩容机制、迭代器失效、增删查改、性能剖析与工程避坑指南
开发语言·c++
为何创造硅基生物17 小时前
独占指针的创建std::make_unique 本身自带堆出现
c++
kyle~18 小时前
ROS 2 与 Isaac Sim 联合仿真(一)体系架构、环境选型与基础通信闭环
c++·机器人·nvidia·仿真·ros2
努力努力再努力wz18 小时前
【内存管理与高并发内存池系列】从 mmap 到 malloc:文件映射、匿名映射与 glibc 内存分配机制详解
linux·c语言·数据结构·数据库·c++·qt·链表
八解毒剂18 小时前
数据结构-平衡二叉树——对二叉搜索树的优化
数据结构·c++·算法
起床困难户57519 小时前
条款20:协助完成返回值优化
c++
啦啦啦啦啦zzzz19 小时前
算法总结(二分查找、双指针)
c++·算法