合并k个升序链表
Tips
题目
将多个链表合并成一个, 并保证新生成的链表按从小到大排序.
提示:
k == lists.length
0 <= k <= 10⁴
0 <= lists[i].length <= 500
-10⁴ <= lists[i][j] <= 10⁴
lists[i]
按升序排列lists[i].length
的总和不超过10⁴
示例
输入: [1 -> 4 -> 5, 1 -> 3 -> 4, 2 -> 6]
输出: 1 -> 1 -> 2 -> 3 -> 4 -> 4 -> 5 -> 6
题解
这道题让我们合并 k
个有序链表, 但是不管合并几个, 基本还是要两两合并.
那么首先考虑的方法是能不能利用 21. 合并两个有序链表 的解法来解答此题. 答案是肯定的, 但是需要修改, 怎么修改呢, 最先想到的就是两两合并, 就是前两个先合并, 合并好了再跟第三个, 然后第四个直到第 k
个. 这样的思路是对的, 但是效率不高, 所以只能换一种思路.
这里就需要用到分治法(Divide and Conquer
). 简单来说就是不停的对半划分, 比如合并 6
个链表, 那么按照分治法, 首先分别合并索引为 0
和 3
, 1
和 4
, 2
和 5
; 这样下一次只需合并 3
个链表, 即合并索引 0
和 2
; 最后和 1
合并就可以了.
代码中的 k
是通过 (n + 1) / 2
计算的, 这里为啥要加 1
呢, 这是为了当 n
为奇数的时候, k
能始终从后半段开始, 比如当 n = 5
时, 那么此时 k = 3
, 则索引 0
和 3
合并, 1
和 4
合并, 最中间的 2
空出来. 当 n
是偶数的时候, 加 1
也不会有影响, 比如当 n = 4
时, 此时 k = 2
, 那么索引 0
和 2
合并, 1
和 3
合并, 参见代码如下:
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode[]} lists
* @return {ListNode}
*/
var mergeKLists = function (lists) {
let n = lists.length
if (n === 0) return null
while (n > 1) {
const k = ((n + 1) / 2) | 0
for (let i = 0; i < ((n / 2) | 0); i++) {
lists[i] = mergeTwoLists(lists[i], lists[i + k])
}
n = k
}
return lists[0]
}
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode} l1
* @param {ListNode} l2
* @return {ListNode}
*/
// 这个就是 21 题原题
var mergeTwoLists = function (l1, l2) {
const dummy = new ListNode(-1)
let curr = dummy
while (l1 && l2) {
if (l1.val < l2.val) {
curr.next = l1
l1 = l1.next
} else {
curr.next = l2
l2 = l2.next
}
curr = curr.next
}
if (!l1) curr.next = l1
if (!l2) curr.next = l2
return dummy.next
}