[LeetCode] 023. Merge k Sorted Lists (Hard) (C++/Java/Python)
來源:程序員人生 發(fā)布時(shí)間:2015-04-07 08:29:39 閱讀次數(shù):3062次
索引:[LeetCode] Leetcode 題解索引 (C++/Java/Python/Sql)
Github:
https://github.com/illuz/leetcode
023. Merge k Sorted Lists (Hard)
鏈接:
題目:https://oj.leetcode.com/problems/merge-k-sorted-lists/
代碼(github):https://github.com/illuz/leetcode
題意:
和
021. Merge Two Sorted Lists (Easy) 類似,這次要 Merge K 個(gè)。
分析:
很明顯可以想到利用已完成的 Merge Two Sorted Lists 的函數(shù)來用。
這時(shí)候有兩種方法:
1. (C++) 用2分的思想,把每一個(gè) List 和它相鄰的 List 進(jìn)行 Merge,這樣范圍就縮小了1半了,再重復(fù)這樣,就能夠 O(nklogk) 完成。比如: [1, 2, …, n] 的第1輪 Merge 是 [1, n/2], [2, n/2+1], …
2. (Python) 也是用2分的思想,就是把 Lists 分為兩部份,分別遞歸 Merge k Sorted Lists 后變成兩個(gè) List ,然后再對(duì)這兩 List 進(jìn)行 Merge Two Sorted Lists 。
這兩種方法都是遞歸調(diào)用,都可以進(jìn)行記憶化,用空間換時(shí)間,不過我不清楚會(huì)不會(huì)超空間(Memory Limit Exceed),所以就沒試了~
除用2分的思路,還有更好寫的方法,就是用堆(heap),具體就是用優(yōu)先隊(duì)列(Priority Queue)。
(Java) 先把每一個(gè) List 的第1個(gè)節(jié)點(diǎn)放進(jìn)優(yōu)先隊(duì)列,每次取出隊(duì)列中的最大值節(jié)點(diǎn),再把那個(gè)節(jié)點(diǎn)的 next 放進(jìn)去。
代碼:
C++:
class Solution {
public:
ListNode *mergeKLists(vector<ListNode *> &lists) {
int sz = lists.size();
if (sz == 0)
return NULL;
while (sz > 1) {
int k = (sz + 1) / 2;
for (int i = 0; i < sz / 2; i++)
lists[i] = mergeTwoLists(lists[i], lists[i + k]);
sz = k;
}
return lists[0];
}
ListNode *mergeTwoLists(ListNode *l1, ListNode *l2) {
if (l1 == NULL)
return l2;
if (l2 == NULL)
return l1;
ListNode *start, *p1;
if (l1->val < l2->val) {
p1 = start = l1;
l1 = l1->next;
} else {
p1 = start = l2;
l2 = l2->next;
}
while (l1 != NULL && l2 != NULL) {
if (l1->val < l2->val) {
p1->next = l1;
p1 = l1;
l1 = l1->next;
} else {
p1->next = l2;
p1 = l2;
l2 = l2->next;
}
}
if (l1 != NULL)
p1->next = l1;
else
p1->next = l2;
return start;
}
};
Java:
public class Solution {
public ListNode mergeKLists(List<ListNode> lists) {
Queue<ListNode> heap = new PriorityQueue<ListNode>(new Comparator<ListNode>(){
@Override public int compare(ListNode l1, ListNode l2) {
return l1.val - l2.val;
}
});
ListNode dummy = new ListNode(0), cur = dummy, tmp;
for (ListNode list : lists) {
if (list != null) {
heap.offer(list);
}
}
while (!heap.isEmpty()) {
tmp = heap.poll();
cur.next = tmp;
cur = cur.next;
if (tmp.next != null) {
heap.offer(tmp.next);
}
}
return dummy.next;
}
}
Python:
class Solution:
# @param a list of ListNode
# @return a ListNode
def mergeKLists(self, lists):
if len(lists) == 0:
return None
if len(lists) == 1:
return lists[0]
mid = len(lists) // 2
left = self.mergeKLists(lists[:mid])
right = self.mergeKLists(lists[mid:])
# merge left and right
dummy = ListNode(0)
cur = dummy
while left or right:
if right == None or (left and left.val <= right.val):
cur.next = left
left = left.next
else:
cur.next = right
right = right.next
cur = cur.next
return dummy.next
生活不易,碼農(nóng)辛苦
如果您覺得本網(wǎng)站對(duì)您的學(xué)習(xí)有所幫助,可以手機(jī)掃描二維碼進(jìn)行捐贈(zèng)