LeetCode 15 3Sum 解题思路和python代码

题目:

Given an integer array nums, return all the triplets nums\[i, numsj, numsk] such that i != j, i != k, and j != k, and numsi + numsj + numsk == 0.

Notice that the solution set must not contain duplicate triplets.

Example 1:

Input: nums = -1,0,1,2,-1,-4

Output: \[-1,-1,2,-1,0,1]

Explanation:

nums0 + nums1 + nums2 = (-1) + 0 + 1 = 0.

nums1 + nums2 + nums4 = 0 + 1 + (-1) = 0.

nums0 + nums3 + nums4 = (-1) + 2 + (-1) = 0.

The distinct triplets are -1,0,1 and -1,-1,2.

Notice that the order of the output and the order of the triplets does not matter.

Example 2:

Input: nums = 0,1,1

Output: \[\]

Explanation: The only possible triplet does not sum up to 0.

Example 3:

Input: nums = 0,0,0

Output: \[0,0,0]

Explanation: The only possible triplet sums up to 0.

Constraints:

3 <= nums.length <= 3000

-105 <= numsi <= 105

解题思路:

首先对数组进行排序,有助于使用双指针方法快速找到合适的三元组。

对于每个元素nums[i],将它作为第一个数,然后使用双指针方法在 i 之后的数组部分寻找另外两个数,使它们的和等于0。

左指针从 i+1 开始,右指针从数组末尾开始,逐步向中间收缩。根据三者之和的结果,调整左右指针的位置。

为了避免重复的三元组,遍历时需要跳过相同的元素。

python 复制代码
class Solution(object):
    def threeSum(self, nums):
        """
        :type nums: List[int]
        :rtype: List[List[int]]
        """
        result = []
        nums.sort()

        print("nums: ", nums)

        for i in range(len(nums)):
            print("i: ", i)

            # 如果当前元素和前一个相同,跳过,避免重复三元组
            if i > 0 and nums[i] == nums[i-1]:
                continue
            
            left, right = i+1, len(nums)-1

            while left < right:
                print("nums[i]: ", nums[i])
                print("nums[left]: ", nums[left])
                print("nums[right]: ", nums[right])
                total = nums[i] + nums[left] + nums[right]
                print("total: ", total)

                if total == 0:
                    result.append([nums[i], nums[left], nums[right]])

                    while left < right and nums[left] == nums[left+1]:
                        left+=1
                    while left < right and nums[right] == nums[right-1]:
                        right-=1
                    
                    left+=1
                    right-=1
                
                elif total < 0:
                    left += 1
                else:
                    right -= 1
        
        return result

例子:

nums = -1,0,1,2,-1,-4

首先,对数组进行排序。

('nums: ', -4, -1, -1, 0, 1, 2)

接着,遍历数组。

('i: ', 0)

('numsi: ', -4)

('numsleft: ', -1)

('numsright: ', 2)

('total: ', -3)

和小于0,说明左指针太小,向右移动左指针。

('numsi: ', -4)

('numsleft: ', -1)

('numsright: ', 2)

('total: ', -3)

和还是小于0,说明左指针太小,向右移动左指针。

('numsi: ', -4)

('numsleft: ', 0)

('numsright: ', 2)

('total: ', -2)

和还是小于0,说明左指针太小,向右移动左指针。

('numsi: ', -4)

('numsleft: ', 1)

('numsright: ', 2)

('total: ', -1)

和还是小于0,但 i=0 的遍历结束,进入 i=1。

('i: ', 1)

('numsi: ', -1)

('numsleft: ', -1)

('numsright: ', 2)

('total: ', 0)

和等于0,将三元组加入result。检查左指针指向的数(-1)是否与下一个数(0)相等,如果相等,就要跳过重复的左指针元素;同样道理,检查右指针指向的数(2)是否与前一个数(1)相等,如果相等,就跳过重复的右指针元素。

移动指针后,继续计算和。

('numsi: ', -1)

('numsleft: ', 0)

('numsright: ', 1)

('total: ', 0)

和等于0,将三元组加入result。

左指针大于右指针,遍历结束。

('i: ', 2)

nums2还是 -1,因此跳过。

('i: ', 3)

('numsi: ', 0)

('numsleft: ', 1)

('numsright: ', 2)

('total: ', 3)

左指针大于右指针,遍历结束。

('i: ', 4)

左指针等于右指针,while loop不会开始。

('i: ', 5)

左指针大于右指针,while loop不会开始。

output是 \[-1,-1,2,-1,0,1]

time complexity是O(n^2)

相关推荐
金銀銅鐵7 小时前
[Python] 基于欧几里得算法,实现分数约分计算器
python·数学
Lyn_Li8 小时前
Kaggle Top 5 | 198只股票、200条数据的金融预测——BattleFin高分方案从零复现
python·kaggle·比赛复盘·金融预测
先吃饱再说11 小时前
判断回文字符串,从一行代码到双指针优化
算法
小九九的爸爸13 小时前
前端想要入门Agent开发,要具备哪些Python基础?
python·agent·ai编程
黄敬峰14 小时前
深入理解算法核心:从递归思想、数组扁平化到快速排序
算法
阿耶同学14 小时前
手把手教你用 LangGraph 搭建三层嵌套 Agent 架构
python·程序员
得物技术15 小时前
从狂野代码到按目标生产:得物推荐 AI Harness 的工程化实践|AICon 演讲整理
人工智能·算法·架构
AI小老六18 小时前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术19 小时前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试