• 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自动化测试等,微信搜索公众号:“无量测试之道”,或扫描下方二维码:


    添加关注,让我们一起共同成长!

  • 相关阅读:
    前端基础学习分享
    IOS中block和代理
    最近一个项目的反思
    Coreseek Windows下安装调试
    iOS对象属性详解
    我眼里“维多利亚的秘密”中最性感的营销三点式
    用CSS制作带图标的按钮
    简单权限管理设计
    CentOS工作内容(三)配置网络IP地址
    CentOS工作内容(二)关闭SELinux
  • 原文地址:https://www.cnblogs.com/Wu13241454771/p/15157398.html
Copyright © 2020-2023  润新知