leetcode第23题

简介: 时间复杂度:假设 N 是所有的数字个数,存到数组是 O(N),排序如果是用快速排序就是 O(Nlog_N)O(NlogN) ,存到链表是 O(N),所以取个最大的,就是 O(Nlog_N)O(NlogN)。空间复杂度:新建了一个链表,O(N)。

image.png

k 个有序链表的合并。

我们用 N 表示链表的总长度,考虑最坏情况,k 个链表的长度相等,都为 n 。

解法一 暴力破解

简单粗暴,遍历所有的链表,将数字存到一个数组里,然后用快速排序,最后再将排序好的数组存到一个链表里。

publicListNodemergeKLists(ListNode[] lists) {
List<Integer>l=newArrayList<Integer>();
//存到数组for (ListNodeln : lists) {
while (ln!=null) {
l.add(ln.val);
ln=ln.next;
        }
    }
//数组排序Collections.sort(l);
//存到链表ListNodehead=newListNode(0);
ListNodeh=head;
for (inti : l) {
ListNodet=newListNode(i);
h.next=t;
h=h.next;
    }
h.next=null;
returnhead.next;
}

时间复杂度:假设 N 是所有的数字个数,存到数组是 O(N),排序如果是用快速排序就是 O(Nlog_N)O(NlogN) ,存到链表是 O(N),所以取个最大的,就是 O(Nlog_N)O(NlogN)

空间复杂度:新建了一个链表,O(N)。

解法二 一列一列比较

image.png


我们可以一列一列的比较,将最小的一个存到一个新的链表里。

publicListNodemergeKLists(ListNode[] lists) {
intmin_index=0;
ListNodehead=newListNode(0);
ListNodeh=head;
while (true) {
booleanisBreak=true;//标记是否遍历完所有链表intmin=Integer.MAX_VALUE;
for (inti=0; i<lists.length; i++) {
if (lists[i] !=null) {
//找出最小下标if (lists[i].val<min) {
min_index=i;
min=lists[i].val;
                }
//存在一个链表不为空,标记改完 falseisBreak=false;
            }
        }
if (isBreak) {
break;
        }
//加到新链表中ListNodea=newListNode(lists[min_index].val);
h.next=a;
h=h.next;
//链表后移一个元素lists[min_index] =lists[min_index].next;
    }
h.next=null;
returnhead.next;
}

时间复杂度:假设最长的链表长度是 n ,那么 while 循环将循环 n 次。假设链表列表里有 k 个链表,for 循环执行 k 次,所以时间复杂度是 O(kn)。

空间复杂度:N 表示最终链表的长度,则为 O(N)。

优先队列的运用印象深刻,此外对两两链表的合并,我们仅仅改变了合并的方式就将时间复杂度降低了很多,美妙!

相关文章
|
7月前
leetcode-1518:换酒问题
leetcode-1518:换酒问题
34 0
|
2月前
【LeetCode 02】暴力法总结
【LeetCode 02】暴力法总结
19 1
leetcode第55题
当自己按照 45 题的思路写完的时候,看 Solution 的时候都懵逼了,这道题竟然这么复杂?不过 Solution 把问题抽象成动态规划的思想,以及优化的过程还是非常值得学习的。
leetcode第55题
leetcode第39题
对回溯法又有了更深的了解,一般的架构就是一个大的 for 循环,然后先 add,接着利用递归进行向前遍历,然后再 remove ,继续循环。而解法二的动态规划就是一定要找到递进的规则,开始的时候就想偏了,导致迟迟想不出来。
leetcode第39题
|
存储
leetcode第52题
可以发现对于同一条副对角线,row + col 的值是相等的。 对于同一条主对角线,row - col 的值是相等的。 我们同样可以用一个 bool 型数组,来保存当前对角线是否有元素,把它们相加相减的值作为下标。 对于 row - col ,由于出现了负数,所以可以加 1 个 n,由 [ - 3, 3 ] 转换为 [ 1 , 7 ] 。
leetcode第52题
|
算法
leetcode第45题
时间复杂度:O(n)。 空间复杂度:O(1)。 这里要注意一个细节,就是 for 循环中,i < nums.length - 1,少了末尾。因为开始的时候边界是第 0 个位置,steps 已经加 1 了。如下图,如果最后一步刚好跳到了末尾,此时 steps 其实不用加 1 了。如果是 i < nums.length,i 遍历到最后的时候,会进入 if 语句中,steps 会多加 1 。
109 0
leetcode第45题
|
存储
leetcode第57题
和上一道可以说是一个问题,只不过这个是给一个已经合并好的列表,然后给一个新的节点依据规则加入到合并好的列表。 解法一 对应 56 题的解法一,没看的话,可以先过去看一下。这个问题其实就是我们解法中的一个子问题没看的话,可以先过去看一下。这个问题其实就是我们解法中的一个子问题, 所以直接加过来就行了
105 0
leetcode第57题
leetcode第14题
我们把原来的数组分成两部分,求出左半部分的最长公共前缀,求出右半部分的最长公共前缀,然后求出的两个结果再求最长公共前缀,就是最后的结果了。 求左半部分的最长公共前缀,我们可以继续把它分成两部分,按照上边的思路接着求。然后一直分成两部分,递归下去。 直到该部分只有 1 个字符串,那么最长公共子串就是它本身了,直接返回就可以了。
leetcode第14题
leetcode第13题
解法一 先来一种不优雅的,也就是我开始的想法。就是遍历字符串,然后转换就可以,但同时得考虑 IV,IX 那些特殊情况。
leetcode第13题