题目大意是传入一个链表数组lists,每个链表都由若干个链接的链表结点组成,并且每个链表结点记录一个整数。题目保证传入的链表中的整数按从小到大进行排序。

题目要求我们输出一个新的链表,这个链表中应该包含所有在lists中出现的整数,并且按从小到大排序。


我的思路:

这个问题解法应该很多。我采用的解法是分治算法,灵感来自于归并排序,因为归并排序的递归完成后也会涉及到将两个有序数组重组合并。

首先说明如何重组两个链表为一个有序链表,其思路源自于我们在玩扑克牌时对两堆有序扑克牌进行排序的手法:

1.我们从两叠有序扑克牌(正面向上)中,选取最小的牌面(即两叠扑克牌盖在最上面的两张牌中较小的牌),并加入到手牌中。

2.循环上一步直到有一叠牌被完全取走。

3.将剩下的一叠牌全部按序加入到手牌中。

由于始终选取较小的牌,以及两叠牌的从小到大排序的性质,能保证剩余的牌总是会不小于被选走的手牌。假设两叠牌共n张,由于这个过程中每次都会使得桌上的牌减少1,而减少到0时就会必定结束,因此上面的步骤最多只会执行n次,这n次中每一次都涉及一次比较和取牌(在计算机中是常量时间),因此我们可以认为要合并两个有序牌堆的时间复杂度为O(n)。

combine(A, B):

  ai = 0, bi = 0

  al = A.length, bl = B.length

  result = empty-list

  while(ai < al && bi <  bl)

    if(A[ai] <= B[bi])

      insert A[ai] into result

      ai = ai + 1

    else

      insert B[bi] into result

      bi = bi + 1

  while(ai < al)

    insert A[ai] into result

    ai = ai + 1

  while(bi < bl)

    insert B[bi] into result

    bi = bi + 1

  return result

但是题目已经指出了,k:=lists.length不一定为2。因此我们要通过分治算法将需要合并的链表数减少为2。其思路是先合并下标为0~k/2的链表,在合并k/2~k的链表。最后利用我们上面所说的洗牌算法将两个链表合并。

rec(lists, from, to) //合并lists[from], ... lists[to -  1]为一个链表,并返回合并的链表

  if(from + 1 == to)

    return lists[i]

  half = (from + to) / 2

  part1 = rec(lists, from, half)

  part2 = rec(lists, half, to)

  return combine(part1, part2)

上面就是我们归并算法的完整实现。先说明它能返回正确的结果。当to - from == 1时,此时需要排序的链表只有一个,只要直接返回这个链表就可以了。因此当to - from == 1时这个算法是正确的。假设当to-from<p时(p >1),这个算法能返回正确的结果。那么当to-from=p时,函数将排序lists中from~half段和half~to段的链表委托给递归函数。由于from < half < to,因此half-from<to-from=p,to-half<to-from=p,因此递归函数对于部分的排序返回了正确的结果,分别记为part1,part2,part1中保存了from~half段的合并结果,而part2中保存了half~to段的合并结果。最终利用我们已经说明过的洗牌合并法对合并part1和part2,因此最终得到的结果则是lists[from], ... lists[to -  1]的有序合并结果。这里用了一个关键的想法,即合并顺序不会改变最终结果。利用数学归纳法我们可以得出对于任意to-from>=1,rec函数都能返回正确的结果。

接下来说明这个算法的复杂度。

首先空间复杂度应该是O(n),因为函数递归发生在空间分配之前,因此当函数从下级递归中跳出时,最多持有与lists中保存的整数的拷贝(保存在part1和part2中),在combine函数中又会分配一次空间,但也最多只是part1和part2的长度的加总,因此,rec函数同时占据的空间量不会超过2n=O(n)。再提一下,这里由于没有要求我们不能使用传入的链表,因此假如我们直接通过操作链表进行排序而不分配额外的空间,那么空间复杂度就可以被优化到O(1)。

时间复杂度我们必须展开来看。一种简明的方式是将不同的递归深度区分开来 ,在同一递归深度中的时间复杂度为O(n)。比方说在第一级递归,我们对长度为k的lists进行合并,但是实际上发生在这一递归深度的只有调用combine的合并操作,其时间复杂度为O(n)。同样在第二级递归,我们分别对0~k/2和k/2~k进行排序,而发生在这一递归深度的实际的操作只有combine操作,两个combine操作正好涉及了lists[0~k]中所有的元素,因此时间复杂度也是O(n)。考虑到递归深度最大为log2(k),因此总的时间复杂度为O(n)*log2(k)=O(nlog2(k))。


最后提供一下java代码,13ms AC:

 /**
  * Definition for singly-linked list.
  * public class ListNode {
  *     int val;
  *     ListNode next;
  *     ListNode(int x) { val = x; }
  * }
  */
 public class Solution {

     public ListNode mergeKLists(ListNode[] lists) {
         if (lists.length <= 1) {
             return lists.length == 0 ? null : lists[0];
         }
         return mergeKLists(lists, 0, lists.length);
     }

     public ListNode mergeKLists(ListNode[] lists, int from, int to) {
         if (to == from + 1) {
             return lists[from];
         }
         int half = (from + to) / 2;
         ListNode part1 = mergeKLists(lists, from, half);
         ListNode part2 = mergeKLists(lists, half, to);
         if (part1 == null || part2 == null) {
             return part1 == null ? part2 : part1;
         }
         if (part1.val > part2.val) {
             ListNode tmp = part1;
             part1 = part2;
             part2 = tmp;
         }
         ListNode traceNode1 = part1;
         ListNode traceNode2 = part2;
         ListNode formerNode1 = null;
         ListNode formerNode2 = null;
         while (traceNode1 != null && traceNode2 != null) {

             while (traceNode1 != null && traceNode2 != null && traceNode1.val <= traceNode2.val) {
                 formerNode1 = traceNode1;
                 traceNode1 = traceNode1.next;
             }
             ListNode start = traceNode2;
             while (traceNode1 != null && traceNode2 != null && traceNode2.val <= traceNode1.val) {
                 formerNode2 = traceNode2;
                 traceNode2 = traceNode2.next;
             }
             if (formerNode1 != null) {
                 formerNode1.next = start;
                 formerNode1 = formerNode2;
             }
             if (formerNode2 != null) {
                 formerNode2.next = traceNode1;
                 formerNode2 = null;
             }
         }
         return part1;
     }
 }

Leetcode:Merge k Sorted Lists分析和实现的更多相关文章

  1. LeetCode: Merge k Sorted Lists 解题报告

    Merge k Sorted Lists Merge k sorted linked lists and return it as one sorted list. Analyze and descr ...

  2. [LeetCode] Merge k Sorted Lists 合并k个有序链表

    Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. 这 ...

  3. [Leetcode] Merge k sorted lists 合并k个已排序的链表

    Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. 思 ...

  4. LeetCode:Merge k Sorted Lists

    题目链接 Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexi ...

  5. LeetCode——Merge k Sorted Lists

    Discription: Merge k sorted linked lists and return it as one sorted list. Analyze and describe its ...

  6. leetcode -- Merge k Sorted Lists add code

    Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity. [ ...

  7. LeetCode Merge k Sorted Lists (链表)

    题意 Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity ...

  8. LeetCode Merge k Sorted Lists 解决报告

    https://oj.leetcode.com/problems/merge-k-sorted-lists/ 归并K已经整理阵列,和分析算法的复杂. 解决报告:无论是不考虑优化,最简单的实现是要重新走 ...

  9. LeetCode —— Merge k Sorted Lists

    /* ** 算法的思路: ** 1.将k个链表的首元素进行建堆 ** 2.从堆中取出最小的元素,放到链表中 ** 3.如果取出元素的有后续的元素,则放入堆中,若没有则转步骤2,直到堆为空 */ #in ...

随机推荐

  1. 使用BloomFilter布隆过滤器解决缓存击穿、垃圾邮件识别、集合判重

    Bloom Filter是一个占用空间很小.效率很高的随机数据结构,它由一个bit数组和一组Hash算法构成.可用于判断一个元素是否在一个集合中,查询效率很高(1-N,最优能逼近于1). 在很多场景下 ...

  2. 旧书重温:0day2【5】shellcode变形记

    紧接上一篇,结合第一篇 //这篇文章主要成功溢出一个带有缓冲区溢出的小程序,其中我们的shellcode被strcpy截断了所以我们需要变形shellcode,这个实验中也出现了很多意想不到的拦路虎, ...

  3. 人生苦短之我用Python篇(遍历、函数、类)

    #遍历 info = {'key1':'value1','key2':'value2','key3':'value3'} #方式一 for i in info: print(i,info[i]) #方 ...

  4. 暴力破解Windows RDP(3389)

    RDP是远程桌面协议. $ nmap your_target Starting Nmap 7.01 ( https://nmap.org ) at 2016-09-20 17:29 CST Nmap ...

  5. ORM 模型层

    一个模型就是一个单独的,确定的数据的信息源,包含了数据的字段和操作方法.通常,每个模型映射为一张数据库中的表 基本原则: 每个模型在Django中的存在形式为一个python类 每个模型都是djang ...

  6. HDU - 6433: H. Pow (简答题,输出大数)

    There are n numbers 3^0, 3^1, . . . , 3^n-1. Each time you can choose a subset of them (may be empty ...

  7. git撤销各种状态下的操作

    使用Git时会出现各种各样的问题,下面是几种情况下怎么反悔的操作 一,未加入缓存区,撤销文件修改 git checkout -- file 二,已加入缓存区,撤销文件提交 git reset HEAD ...

  8. 【HDU】4632 Palindrome subsequence(回文子串的个数)

    思路:设dp[i][j] 为i到j内回文子串的个数.先枚举所有字符串区间.再依据容斥原理. 那么状态转移方程为   dp[i][j] = dp[i][j-1] + dp[i+1][j] - dp[i+ ...

  9. envoy 测试试用

    备注: 为了简单测试使用的是docker 镜像进行的测试   1.  Dockerfile   FROM lyft/envoy:latest RUN apt-get update COPY envoy ...

  10. Maven项目中突然找不到Build Path或maven dependencies library

    这两天发现有个maven项目抽风了,一个是右击项目找不到Build Path了,一个是依赖的lib库没了,maven引入的依赖包导不了.后来发现是eclipse搞的鬼,出问题的是项目下的.classp ...