LeetCode 面试经典 150 题-分治

LeetCode 面试经典 150 题-分治

将有序数组转换为二叉搜索树

分治法

如果只有一个节点,直接作为根;前面的部分是左子树,后面的部分是右子树。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class Solution {
public TreeNode sortedArrayToBST(int[] nums) {
return dfs(nums, 0, nums.length);
}

// 把left-right-1转换成平衡二叉搜索树
private TreeNode dfs(int[] nums, int left, int right) {
// 递归结束
if (left == right) {
return null;
}
// 一分为二
int mid = (left + right) >> 1;
return new TreeNode(nums[mid], dfs(nums, left, mid), dfs(nums, mid + 1, right));
}
}
  • 时间复杂度:O(n)
  • 空间复杂度:O(n)

排序链表

自顶向下归并排序

  1. 找到链表的中点,以中点为分界,将链表拆分成两个子链表。寻找链表的中点可以使用快慢指针的做法,快指针每次移动 2 步,慢指针每次移动 1 步,当快指针到达链表末尾时,慢指针指向的链表节点即为链表的中点。
  2. 对两个子链表分别排序。
  3. 将两个排序后的子链表合并,得到完整的排序后的链表。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode sortList(ListNode head) {
// 如果链表为空或者只有一个节点,无需排序
if (head == null || head.next == null) {
return head;
}
// 找到中间节点,并断开与其前一个节点的连接
ListNode pre = middleNode(head);
ListNode mid = pre.next;
pre.next = null;
// 分治
head = sortList(head);
mid = sortList(mid);
// 合并
return merge(head, mid);
}

public ListNode middleNode(ListNode head) {
ListNode slow = head;
ListNode fast = head;
while (fast.next != null && fast.next.next != null) {
slow = slow.next;
fast = fast.next.next;
}
return slow;
}

// 合并两个有序链表
public ListNode merge(ListNode left, ListNode right) {
ListNode dummy = new ListNode();
ListNode cur = dummy;
while (left != null && right != null) {
if (left.val < right.val) {
cur.next = left;
left = left.next;
} else {
cur.next = right;
right = right.next;
}
cur = cur.next;
}
// 链接剩余部分
cur.next = left != null ? left : right;
return dummy.next;
}
}
  • 时间复杂度:O(nlogn)
  • 空间复杂度:O(logn)

自底向上归并排序

  1. 遍历链表,获取链表长度 length。
  2. 初始化步长 step=1。
  3. 循环直到 step≥length。
  4. 每轮循环,从链表头节点开始。
  5. 分割出两段长为 step 的链表,合并,把合并后的链表插到新链表的末尾。重复该步骤,直到链表遍历完毕。
  6. 把 step 扩大一倍。回到第 4 步。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode sortList(ListNode head) {
// 如果链表为空或只有一个元素,直接返回
if (head == null || head.next == null) {
return head;
}

// 获取链表长度
int len = getLength(head);

// 从小到大逐渐合并
ListNode dummy = new ListNode(0);
dummy.next = head;

// 每次合并两个相邻的部分,部分的大小是step
for (int step = 1; step < len; step *= 2) {
ListNode cur = dummy.next;
ListNode tail = dummy;

// 每次处理一组大小为step的链表
while (cur != null) {
// 分割出两个部分
ListNode left = cur;
ListNode right = split(left, step);
cur = split(right, step); // cur 指向下一个部分

// 合并这两部分
tail.next = merge(left, right);

// 移动 tail 指针到合并后的末尾
while (tail.next != null) {
tail = tail.next;
}
}
}

return dummy.next;
}

// 获取链表长度
private int getLength(ListNode head) {
int length = 0;
while (head != null) {
length++;
head = head.next;
}
return length;
}

// 分割链表
private ListNode split(ListNode head, int size) {
if (head == null) return null;
for (int i = 1; i < size && head.next != null; i++) {
head = head.next;
}
ListNode second = head.next;
head.next = null;
return second;
}

// 合并两个有序链表
private ListNode merge(ListNode left, ListNode right) {
ListNode dummy = new ListNode(0);
ListNode cur = dummy;

while (left != null && right != null) {
if (left.val < right.val) {
cur.next = left;
left = left.next;
} else {
cur.next = right;
right = right.next;
}
cur = cur.next;
}

// 连接剩余部分
if (left != null) {
cur.next = left;
} else {
cur.next = right;
}

return dummy.next;
}
}
  • 时间复杂度:O(nlogn)
  • 空间复杂度:O(1)

建立四叉树

这道题以后再做。

合并k个有序链表

分治法

将链表数组分治成较小的子问题。每次通过中点将数组分成两部分,分别对这两部分递归调用 mergeSort,直到数组只剩下一个链表为止。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode mergeKLists(ListNode[] lists) {
// 边界处理
int k = lists.length;
if (k == 0) {
return null;
}
return mergeSort(lists, 0, k - 1);
}

public ListNode mergeSort(ListNode[] lists, int left, int right) {
// 递归结束
if (left == right) {
return lists[left];
}
// 递归合并
int mid = (left + right) >> 1;
return merge(mergeSort(lists, left, mid), mergeSort(lists, mid + 1, right));
}

public ListNode merge(ListNode list1, ListNode list2) {
ListNode dummy = new ListNode();
ListNode cur = dummy;

while(list1 != null && list2 != null) {
if (list1.val < list2.val) {
cur.next = list1;
list1 = list1.next;
} else {
cur.next = list2;
list2 = list2.next;
}
cur = cur.next;
}
cur.next = list1 != null ? list1 : list2;

return dummy.next;
}
}
  • 时间复杂度:O(n * k * log(k))
  • 空间复杂度:O(logk)

最小堆

维护当前每个链表没有被合并的元素的最前面一个,k 个链表就最多有 k 个满足这样条件的元素,每次在这些元素里面选取 val 属性最小的元素合并到答案中。在选取最小元素的时候,可以用优先队列来优化这个过程。

初始把所有链表的头节点入堆,然后不断弹出堆中最小节点 x,如果 x.next 不为空就加入堆中。循环直到堆为空。把弹出的节点按顺序拼接起来,就得到了答案。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
/**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution {
public ListNode mergeKLists(ListNode[] lists) {
// 初始化最小堆
PriorityQueue<ListNode> pq = new PriorityQueue<>((a, b) -> a.val - b.val);
for (ListNode node : lists) {
if (node != null) {
pq.offer(node);
}
}

// 合并链表
ListNode dummy = new ListNode();
ListNode cur = dummy;
while (!pq.isEmpty()) {
// 最小堆中的最小节点
ListNode node = pq.poll();
if (node.next != null) {
pq.offer(node.next);
}
// 合并到链表中
cur.next = node;
cur = cur.next;
}

return dummy.next;
}
}
  • 时间复杂度:考虑优先队列中的元素不超过 k 个,那么插入和删除的时间代价为 O(logk),这里最多有 kn 个点,对于每个点都被插入删除各一次,故总的时间代价即渐进时间复杂度为 O(kn×logk)。
  • 空间复杂度:O(k),优先队列中的元素不超过 k 个。

LeetCode 面试经典 150 题-分治
http://example.com/2025/02/25/posts/hot150-12/
作者
Xuan Yang
发布于
2025年2月25日
许可协议