牛客网高频算法题系列-BM5-合并k个已排序的链表

题目描述

合并 k 个升序的链表并将结果作为一个升序的链表返回其头节点。

原题目见:BM5 合并k个已排序的链表

解法一:分治法

分治法,可以将大问题分解成小问题,然后继续分解成最小的子问题并解决之。

具体处理过程如下,将k个链表分解成2部分处理,递归处理这2部分,并调用 BM4 合并两个排序的链表 中的方法将2个合并好的链表进行合并,最小的子问题的条件是:

  • 没有待合并的链表,直接返回空。
  • 如果只有一个链表,则不需要合并,直接返回该链表。

如果不满足,则需要继续分解并递归处理。

说明:BM4 合并两个排序的链表,请参考 牛客网高频算法题系列-BM4-合并两个排序的链表。

代码

  1. import com.google.common.collect.Lists;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. public class Bm005 {
  5. /**
  6. * 分治法
  7. *
  8. * @param lists
  9. * @return
  10. */
  11. public static ListNode mergeKLists(List<ListNode> lists) {
  12. // 如果没有待合并的链表,直接返回空
  13. if (lists == null || lists.size() == 0) {
  14. return null;
  15. }
  16. // 如果只有一个链表,则不需要合并,直接返回该链表
  17. if (lists.size() == 1) {
  18. return lists.get(0);
  19. }
  20. int left = 0, right = lists.size();
  21. int mid = (left + right) / 2;
  22. // 递归调用当前方法将原有的链表集合分成2部分分别进行合并
  23. // 然后调用 BM4 合并两个排序的链表 中的方法将2个合并好的链表进行合并
  24. return Bm004.merge(mergeKLists(lists.subList(0, mid)), mergeKLists(lists.subList(mid, right)));
  25. }
  26. public static void main(String[] args) {
  27. ListNode listNode1 = ListNode.testCase3();
  28. System.out.println("链表一");
  29. ListNode.print(listNode1);
  30. ListNode listNode2 = ListNode.testCase4();
  31. System.out.println("链表二");
  32. ListNode.print(listNode2);
  33. ListNode listNode3 = new ListNode(7);
  34. System.out.println("链表三");
  35. ListNode.print(listNode3);
  36. ArrayList<ListNode> lists = Lists.newArrayList(listNode1, listNode2, listNode3);
  37. ListNode result = mergeKLists(lists);
  38. // 测试用例,期望输出: 1 -> 2 -> 3 -> 4 -> 5 -> 6 -> 7
  39. System.out.println("合并后的链表");
  40. ListNode.print(result);
  41. }
  42. }

牛客网高频算法题系列-BM5-合并k个已排序的链表 - 图1
牛客网高频算法题系列-BM5-合并k个已排序的链表 - 图2
相信坚持的力量!