10.三数之和

题目描述:
  • 给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组。
  • 注意:答案中不可以包含重复的三元组。
  • 示例
例如, 给定数组 nums = [-1, 0, 1, 2, -1, -4],

满足要求的三元组集合为:
[
  [-1, 0, 1],
  [-1, -1, 2]
]
python代码:

思路1:暴力解法

class Solution:
    def threeSum(self, nums):
        result = []
        n = len(nums)
        for i in range(n):
            tmp_num1 = 0-nums[i]    #tmp_num1与tmp_num2不能使用相同变量名
            for j in range(i+1, n):
                tmp_num2 = tmp_num1 - nums[j]
                for k in range(j+1, n):
                    if nums[k] == tmp_num2:
                        li = [nums[i], nums[j], nums[k]]
                        li.sort()
                        if li not in result:   #去重
                            result.append(li)            
        return result

思路2:

  • 先将给定 nums 排序。
  • 找组合思路:固定三个数字中最左数字的指针 k,遍历 k 找到每个k对应的所有满足nums[k] + nums[i] + nums[j] == 0的 i,j 组合。即每指向新的 nums[k],都通过双指针法找到所有和为 0 的 nums[i],nums[j] 并记录:
  • 当 nums[k] > 0 时直接跳出,因为 nums[k] + nums[i] + nums[j] 大于 0,在此k之后不可能找到组合了;
  • 当 k > 0且nums[k] == nums[k - 1]时跳过此数字k,因为 nums[k - 1] 的所有组合已经被加入到结果中,本次搜索只会搜索到重复组合。
  • i,j 分设在数组 [k, len(nums)] 两端,根据 sum 与 0 的大小关系交替向中间逼近,如果遇到等于 0 的组合则加入 res 中,需要注意:移动 i,j 需要跳过所有重复值,以避免重复答案被计入 res。
class Solution:
    def threeSum(self, nums):
        nums.sort()
        res = []
        for i in range(len(nums) - 2):
            if nums[i] > 0:     # 此时nums[i] + nums[j] + nums[k] > 0
                break
            if i > 0 and nums[i] == nums[i-1]:  # 去重
                continue
            j = i + 1
            k = len(nums) - 1
            while j < k:
                s = nums[i] + nums[j] + nums[k]
                if s > 0:      # 减小s    
                    k -= 1
                    while nums[k] == nums[k+1] and j < k:  #去重
                        k -= 1
                elif s < 0:    # 增大s
                    j += 1
                    while nums[j] == nums[j-1] and j < k:  #去重
                        j += 1
                else:
                    res.append([nums[i], nums[j], nums[k]])  #保存
                    j += 1
                    k -= 1
                    while nums[k] == nums[k+1] and j < k:  #去重
                        k -=1
                    while nums[j] == nums[j-1] and j < k:  #去重
                        j += 1
        return res
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值