hot 100 第三十三 33.排序链表

给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表

示例 1:

复制代码
输入:head = [4,2,1,3]
输出:[1,2,3,4]

示例 2:

复制代码
输入:head = [-1,5,3,4,0]
输出:[-1,0,3,4,5]

示例 3:

复制代码
输入:head = []
输出:[]

核心思路

归并排序(自底向上):链表的归并排序,自底向上避免递归栈开销。

复制代码
链表: 4 → 2 → 1 → 3

步骤1: 每1个节点一组,两两合并
[4] [2] [1] [3]
 ↓   ↓   ↓   ↓
[2,4] [1,3]

步骤2: 每2个节点一组,两两合并
[2,4] [1,3]
   ↓     ↓
  [1,2,3,4]

结果: 1 → 2 → 3 → 4

题解:

java 复制代码
class Solution {
    public ListNode sortList(ListNode head) {
        if (head == null || head.next == null) return head;
        
        // 1. 用快慢指针找到中点
        ListNode slow = head;
        ListNode fast = head.next;  // fast从head.next开始,确保slow在中点偏左
        
        while (fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
        }
        
        // 2. 断开链表
        ListNode mid = slow.next;
        slow.next = null;
        
        // 3. 递归排序左右两部分
        ListNode left = sortList(head);
        ListNode right = sortList(mid);
        
        // 4. 合并两个有序链表
        return mergeTwoLists(left, right);
    }
    
    private ListNode mergeTwoLists(ListNode l1, ListNode l2) {
        ListNode dummy = new ListNode(0);
        ListNode curr = dummy;
        
        while (l1 != null && l2 != null) {
            if (l1.val <= l2.val) {
                curr.next = l1;
                l1 = l1.next;
            } else {
                curr.next = l2;
                l2 = l2.next;
            }
            curr = curr.next;
        }
        
        curr.next = (l1 != null) ? l1 : l2;
        return dummy.next;
    }
}
```

### 递归演示
```
sortList(4→2→1→3)
  ├─ 找中点: slow=2
  ├─ 断开: [4→2] [1→3]
  ├─ sortList(4→2)
  │   ├─ 找中点: slow=4
  │   ├─ 断开: [4] [2]
  │   ├─ sortList(4) → 4
  │   ├─ sortList(2) → 2
  │   └─ merge(4, 2) → 2→4
  │
  ├─ sortList(1→3)
  │   ├─ 找中点: slow=1
  │   ├─ 断开: [1] [3]
  │   ├─ sortList(1) → 1
  │   ├─ sortList(3) → 3
  │   └─ merge(1, 3) → 1→3
  │
  └─ merge(2→4, 1→3) → 1→2→3→4
```

## 两种解法对比

| 特性 | 自底向上 | 自顶向下 |
|------|---------|---------|
| **时间复杂度** | O(n log n) | O(n log n) |
| **空间复杂度** | O(1) | O(log n) --- 递归栈 |
| **代码难度** | 较难 | 中等 |
| **符合题目要求** | ✓ 完全符合 | 空间不是O(1) |

## 快慢指针找中点详解
```
为什么fast从head.next开始?

链表: 1 → 2 → 3 → 4

如果fast从head开始:
slow = 1, fast = 1
slow = 2, fast = 3
slow = 3, fast = null
slow停在3(中点偏右)

断开: [1→2→3] [4]
左边比右边长,递归不平衡

如果fast从head.next开始:
slow = 1, fast = 2
slow = 2, fast = 4
slow = 2, fast = null
slow停在2(中点偏左)

断开: [1→2] [3→4]
左右平衡 ✓

本质

链表排序的关键:

  1. 选择合适算法 --- 归并排序最适合链表
  2. 自底向上实现 --- 避免递归栈,达到O(1)空间
  3. 链表分割 --- 用cut函数和快慢指针
  4. 合并有序链表 --- 核心作

这道题综合了:

  • 归并排序思想
  • 链表分割技巧
  • 合并有序链表
  • 空间优化方法
相关推荐
灵智实验室13 分钟前
PX4状态估计技术EKF2详解(四):EKF2 Output Predictor——从延迟估计到实时输出
算法·无人机·px 4
科研小白_32 分钟前
【MATLAB点云处理基础】基于区域生长算法的桥墩面域点云分割
算法
paeamecium1 小时前
【PAT甲级真题】- Shuffling Machine (20)
c++·算法·pat考试·pat
m0_737539371 小时前
pod Scheduler调度
算法·贪心算法
此生决int1 小时前
算法从入门到精通——双指针
算法
普马萨特1 小时前
Uber H3:地理网格索引在空间数据分析中的应用
数据结构·算法
alphaTao1 小时前
LeetCode 每日一题 2026/5/11-2026/5/17
算法·leetcode
洛水水1 小时前
【力扣100题】45.零钱兑换
算法·leetcode·职场和发展
Aaron15882 小时前
全频段 SDR干扰源模块解决方案(星链干扰、LORA无人机干扰)
人工智能·算法·fpga开发·硬件架构·硬件工程·无人机·信息与通信
AI科技星2 小时前
全域数学·球面拓扑微扰标准系数η=0.01 应用详解(典籍正式版)
人工智能·算法·数学建模·数据挖掘·机器人