leetcode第15题

简介: 参考了这里-Java-solution)主要思想是,遍历数组,用 0 减去当前的数,作为 sum ,然后再找两个数使得和为 sum。这样看来遍历需要 O(n),再找两个数需要 O(n²)的复杂度,还是需要 O(n³)。巧妙之处在于怎么找另外两个数。最最优美的地方就是,首先将给定的 num 排序。这样我们就可以用两个指针,一个指向头,一个指向尾,去找这两个数字,这样的话,找另外两个数时间复杂度就会从 O(n²),降到 O(n)。而怎么保证不加入重复的 list 呢?要记得我们的 nums 已经有序了,所以只需要找到一组之后,当前指针要移到和当前元素不同的地方。其次在遍

image.png

top15

解法一 暴力解法

无脑搜索,三层循环,遍历所有的情况。但需要注意的是,我们需要把重复的情况去除掉,就是 [1, -1 ,0] 和 [0, -1, 1] 是属于同一种情况的。

publicList<List<Integer>>threeSum(int[] nums) {
List<List<Integer>>res=newArrayList<List<Integer>>();
for (inti=0; i<nums.length; i++) {
for (intj=i+1; j<nums.length; j++)
for (intk=j+1; k<nums.length; k++) {
if (nums[i] +nums[j] +nums[k] ==0) {
List<Integer>temp=newArrayList<Integer>();
temp.add(nums[i]);
temp.add(nums[j]);
temp.add(nums[k]); 
//判断结果中是否已经有 temp 。if (isInList(res, temp)) {
continue;
                    }
res.add(temp);
                }
            }
    }
returnres;
}
publicbooleanisInList(List<List<Integer>>l, List<Integer>a) {
for (inti=0; i<l.size(); i++) {
//判断两个 List 是否相同if (isSame(l.get(i), a)) {
returntrue;
        }
    }
returnfalse;
}
publicbooleanisSame(List<Integer>a, List<Integer>b) {
intcount=0;
Collections.sort(a);
Collections.sort(b);
//排序后判断每个元素是否对应相等for (inti=0; i<a.size(); i++) {
if (a.get(i) !=b.get(i)) {
returnfalse;
        }
    }
returntrue;
}

时间复杂度:n 表示 num 的个数,三个循环 O(n³),而 isInList 也需要 O(n),总共就是 O(n^4)O(n4),leetCode 复杂度到了 O(n^3)O(n3) 一般就报超时错误了,所以算法还得优化。

空间复杂度:最坏情况,即 O(N), N 是指 n 个元素的排列组合个数,即 N=C^3_nN=Cn3,用来保存结果。

解法二

参考了这里-Java-solution)

主要思想是,遍历数组,用 0 减去当前的数,作为 sum ,然后再找两个数使得和为 sum。

这样看来遍历需要 O(n),再找两个数需要 O(n²)的复杂度,还是需要 O(n³)。

巧妙之处在于怎么找另外两个数。

最最优美的地方就是,首先将给定的 num 排序。

这样我们就可以用两个指针,一个指向头,一个指向尾,去找这两个数字,这样的话,找另外两个数时间复杂度就会从 O(n²),降到 O(n)。

而怎么保证不加入重复的 list 呢?

要记得我们的 nums 已经有序了,所以只需要找到一组之后,当前指针要移到和当前元素不同的地方。其次在遍历数组的时候,如果和上个数字相同,也要继续后移。文字表述比较困难,可以先看下代码。

publicList<List<Integer>>threeSum(int[] num) {
Arrays.sort(num); //排序List<List<Integer>>res=newLinkedList<>(); 
for (inti=0; i<num.length-2; i++) {
//为了保证不加入重复的 list,因为是有序的,所以如果和前一个元素相同,只需要继续后移就可以if (i==0|| (i>0&&num[i] !=num[i-1])) {
//两个指针,并且头指针从i + 1开始,防止加入重复的元素intlo=i+1, hi=num.length-1, sum=0-num[i];
while (lo<hi) {
if (num[lo] +num[hi] ==sum) {
res.add(Arrays.asList(num[i], num[lo], num[hi]));
//元素相同要后移,防止加入重复的 listwhile (lo<hi&&num[lo] ==num[lo+1]) lo++;
while (lo<hi&&num[hi] ==num[hi-1]) hi--;
lo++; hi--;
                } elseif (num[lo] +num[hi] <sum) lo++;
elsehi--;
           }
        }
    }
returnres;
}

时间复杂度:O(n²),n 指的是 num

空间复杂度:O(N),最坏情况,即 N 是指 n 个元素的排列组合个数,即 N=C^3_nN=Cn3,用来保存结果。

对于遍历,这里用到了从两头同时遍历,从而降低了时间复杂度,很妙!



相关文章
|
5月前
|
C++ Python
leetcode-283:移动零
leetcode-283:移动零
28 0
|
5月前
|
消息中间件 Kubernetes NoSQL
LeetCode 3、28、1351
LeetCode 3、28、1351
|
索引
leetcode第53题
解法一和解法二的动态规划,只是在定义的时候一个表示以 i 开头的子数组,一个表示以 i 结尾的子数组,却造成了时间复杂度的差异。问题就是解法一中求出了太多的没必要的和,不如解法二直接,只保存最大的和。
leetcode第53题
|
算法
leetcode第47题
基本上都是在上道题的基础上改出来了,一些技巧也是经常遇到,比如先排序,然后判断和前一个是否重复。利用 Hash 去重的功能。利用原来的存储空间隐藏掉数据,然后再想办法还原。
leetcode第47题
leetcode第55题
当自己按照 45 题的思路写完的时候,看 Solution 的时候都懵逼了,这道题竟然这么复杂?不过 Solution 把问题抽象成动态规划的思想,以及优化的过程还是非常值得学习的。
leetcode第55题
leetcode第38题
难在了题目是什么意思呢? 初始值第一行是 1。 第二行读第一行,1 个 1,去掉个字,所以第二行就是 11。 第三行读第二行,2 个 1,去掉个字,所以第三行就是 21。 第四行读第三行,1 个 2,1 个 1,去掉所有个字,所以第四行就是 1211。 第五行读第四行,1 个 1,1 个 2,2 个 1,去掉所有个字,所以第五航就是 111221。 第六行读第五行,3 个 1,2 个 2,1 个 1,去掉所以个字,所以第六行就是 312
leetcode第38题
|
算法
leetcode第40题
会发现出现了很多重复的结果,就是因为没有跳过重复的 1。在求 opt [ 1 ] 的时候就变成了 [ [ 1 ],[ 1 ] ] 这样子,由于后边求的时候都是直接在原来每一个列表里加数字,所有后边都是加了两次了。
leetcode第40题
|
存储
leetcode第56题
常规的思想,将大问题化解成小问题去解决。 假设给了一个大小为 n 的列表,然后我们假设 n - 1 个元素的列表已经完成了全部合并,我们现在要解决的就是剩下的 1 个,怎么加到已经合并完的 n -1 个元素中。 这样的话分下边几种情况, 我们把每个范围叫做一个节点,节点包括左端点和右端点。 1. 如下图,新加入的节点左端点和右端点,分别在两个节点之间。这样,我们只要删除
leetcode第56题