LeetCode.两数之和和三数之和
两数之和
2次for循环O( N2 ) 的做法就不说了,大家都会。我说下O(N)时间复杂度的做法。
解题思路:
一次for循环。遍历的时候用字典记录下遍历到的值,不过是以num为key,以下标index为value
继续遍历的时候,就用 target - num作为key去字典中取值,如果取到了说明存在,返回结果
若没有取到,就以num为key,index为value存入字典
代码如下:
func twoSum(_ nums: [Int], _ target: Int) -> [Int] { var map = [Int: Int]() for (i,num) in nums.enumerated() { if let index = map[target - num] { return [index, i] } map[num] = i } return [] }
三数之和:
其实这个题目有点难,暴力解法就是3层for循环,不过O( N3 )提交是肯定过不了的,这个题目还有一个要求就是不能重复。
解题思路:
先进行一次排序。因为这个题目考查的不是排序,所以我们可以直接使用系统自带的排序算法。
上图中,l的下标一开始是 i+1; r一开始就只想最后一个元素,下标为nums.count-1;
就以-4开始举一个例子来表述这个算法的解题思路:
- 开始时,i = 0, nums[i] = -4, l = i + 1, r = nums.count - 1
- 此时,nums[i] + nums[l] + nums[r] = -4-1+2 = -3,因为数组是排好序的,说明nums[l]太小了,所以l++;
- 直到l == 4,此时,nums[i] + nums[l] + nums[r] = -4+1+2 = -1还是不等于0,所以i++
- 此时i == 1,l = i+1 = 2,r = nums.count - 1,此时nums[i] + nums[l] + nums[r] = -1-1+2 = 0,所以将这组数据存入到结果数组中。然后l++.r--,进行下一次遍历。
代码如下:
class Solution { func threeSum(_ nums: [Int]) -> [[Int]] { if nums.count < 3 { return [] } // 系统方法进行排序 let sortedNums = nums.sorted() var result = [[Int]]() let lastR = sortedNums.count - 1 for i in 0...(sortedNums.count - 3) { // 去重1⃣️ if i > 0 && sortedNums[i] == sortedNums[i-1] { continue } var l = i + 1; var r = lastR; let remain = 0 - sortedNums[i]; while (l < r) { let sumLR = sortedNums[l] + sortedNums[r] if sumLR == remain { result.append( [sortedNums[i], sortedNums[l], sortedNums[r] ]) // 去重2⃣️ while (l < r && sortedNums[l] == sortedNums[l+1]) { l += 1 } while (l < r && sortedNums[r] == sortedNums[r-1]) { r -= 1 } l += 1 r -= 1 } else if sumLR < remain { l += 1 } else { r -= 1 } } } return result } }
核心是如何去重的问题。举例子来说明重复的问题,代码里面有2处重复,我标记了分别为1⃣️和2⃣️。
先看第1⃣️处去重情况:看下图红色方框选中的两个-1
。因为前面一个-1
已经完成一次遍历。所以再碰到同样的 -1
就直接跳过。
再看第2⃣️处去重情况:如下图所示,当i = 0, l = 2, r = 6时,nums[i] + nums[l] + nums[r] = 0,所以马上l++, r--红色的表示转换后的样子,发现l++ 和 l指的指向是一样的,所以就跳过。r 和 r--指的指向是一样的,也跳过。这样就达到去重的目的。
总结:
今天主要分享了2道常考的算法题。2数之和和3数之和。2数之和的思路就是用字典来存储,用空间换时间的方式使时间复杂度为O(N)。3数之和就比较有难度了,核心就是如何去重的问题,我也在文中举例子来表达了是如何去重的。希望能帮助到大家。
欢迎关注【无量测试之道】公众号,回复【领取资源】
Python编程学习资源干货、
Python+Appium框架APP的UI自动化、
Python+Selenium框架Web的UI自动化、
Python+Unittest框架API自动化、
资源和代码 免费送啦~
文章下方有公众号二维码,可直接微信扫一扫关注即可。
备注:我的个人公众号已正式开通,致力于测试技术的分享,包含:大数据测试、功能测试,测试开发,API接口自动化、测试运维、UI自动化测试等,微信搜索公众号:“无量测试之道”,或扫描下方二维码:
添加关注,让我们一起共同成长!