专题1 - 双指针 - leetcode 15. 三数之和 - 中等难度

leetcode 15. 三数之和 - 点击直达

  • leetcode 15. 三数之和 中等难度 双指针
    • 1. 题目详情
      • 1. 原题链接
      • 2. 基础框架
    • 2. 解题思路
      • 1. 题目分析
      • 2. 算法原理
      • 3. 时间复杂度
    • 3. 代码实现
    • 4. 知识与收获

在这里插入图片描述

leetcode 15. 三数之和 中等难度 双指针

1. 题目详情

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != j、i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请

你返回所有和为 0 且不重复的三元组。

注意:答案中不可以包含重复的三元组。

示例 1:
输入:nums = [-1,0,1,2,-1,-4]
输出:[[-1,-1,2],[-1,0,1]]
解释:
nums[0] + nums[1] + nums[2] = (-1) + 0 + 1 = 0 。
nums[1] + nums[2] + nums[4] = 0 + 1 + (-1) = 0 。
nums[0] + nums[3] + nums[4] = (-1) + 2 + (-1) = 0 。
不同的三元组是 [-1,0,1] 和 [-1,-1,2] 。
注意,输出的顺序和三元组的顺序并不重要。

示例 2:
输入:nums = [0,1,1]
输出:[]
解释:唯一可能的三元组和不为 0 。

示例 3:
输入:nums = [0,0,0]
输出:[[0,0,0]]
解释:唯一可能的三元组和为 0 。

提示:
3 <= nums.length <= 3000
-105 <= nums[i] <= 105

1. 原题链接

leetcode 15. 三数之和

2. 基础框架

● Cpp代码框架

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        
    }
};

2. 解题思路

1. 题目分析

( 1 ) (1) (1) 本题要求找出数组 n u m s nums nums中满足条件:三个位置都不同数 n u m s [ i ] + n u m s [ j ] + n u m s [ k ] = = t a r g e t nums[i]+nums[j]+nums[k]==target nums[i]+nums[j]+nums[k]==target的所有结果。

2. 算法原理

( 1 ) (1) (1) 首先想到暴力解法:先进行三层for循环遍历数组 n u m s nums nums,然后使用C++中set容器对结果进行去重处理,时间复杂度是 O ( n 3 ) O(n^3) O(n3)

( 2 ) (2) (2) 在分析三数之和之前,我们先来看看两数之和为target时,如何在nums中寻找这两个数:
首先还是想到暴力解法:进行两层for循环,时间复杂度 O ( n 2 ) O(n^2) O(n2),如何优化呢?

使用对撞双指针算法:
先看张图吧:
在这里插入图片描述

先对数组 n u m s nums nums进行快速排序, n u m s nums nums中就得到非递减的序列;
有序数组好就好在有序上,我们就可以找规律了:
左指针left指向第一个元素,右指针right指向最后一个元素。此时两个数的和 s u m = n u m s [ l e f t ] + n u m s [ r i g h t ] sum = nums[left] + nums[right] sum=nums[left]+nums[right] 注意这是一个有序数组,left指向的元素是[left, right]范围内最小的元素,right指向的元素是[left, right]中最大的元素。
sum的变化情况是:left向右移动时,sum不变或增大;right向左移动时sum不变或减小;
所以我们可以通过比较sumtarget(本题target是0)的大小来确定移动的情况:
1.sum > target,想要趋近targetsum需要减小,所以移动右指针right
2.sum < target,想要趋近targetsum需要增大,所以移动左指针left
3.sum == target,得到了一个结果nums[left]和nums[right]。但是题目要求得到所有的结果,所以还需要继续,此时为了保证结果不重复,需要依据当前得到的结果进行去重处理(即当前已经有了结果nums[left],之后的出现的所有与nums[left]相同的元素都直接略过不在考虑,因为一定是结果中有的。right同理)
本题去重就是控制left和right跳过所有与自身重复的元素,这里需要注意的是控制left++或right--时需要先判断left或right是否越界,因为在极端情况下所有元素都相同,指针一直移动导致越界(细节细节)
在这里插入图片描述

( 3 ) (3) (3) 好了,现在我们来看三数之和,其实就是上文两数之和的变形。三数之和的题目要求 n u m s [ i ] + n u m s [ l e f t ] + n u m s [ r i g h t ] = = t a r g e t nums[i]+nums[left]+nums[right]==target nums[i]+nums[left]+nums[right]==target本题target为0),变形为 n u m s [ l e f t ] + n u m s [ r i g h t ] = = t a r g e t − n u m s [ k ] nums[left]+nums[right]==target-nums[k] nums[left]+nums[right]==targetnums[k],把target-nums[i]整体当做新的target,记为newtarget,所以得到 n u m s [ l e f t ] + n u m s [ r i g h t ] = = n e w t a r g e t nums[left]+nums[right] == newtarget nums[left]+nums[right]==newtarget
在这里插入图片描述

( 4 ) (4) (4) 具体做法就是在两数之和的基础上在加上一层循环,遍历数组nums,每次确定一个nums[i],即确定一个newtarget。内层双指针leftright在范围[i+1, n-1]内以newtarget为目标进行寻找。
在本次循环结束时需要额外在做的一点是对nums[i]也要进行进行去重处理,同时也需要进行是否越界访问的判断(细节细节),至于为什么要进行去重,原理和两数之和做法类似。

( 5 ) (5) (5)

3. 时间复杂度

O ( n 2 ) O(n^2) O(n2)

外层循环每次确定一个目标值,内层循环在目标值之后的区间内寻找满足两数之和条件的结果。

3. 代码实现

class Solution {
public:
    vector<vector<int>> threeSum(vector<int>& nums) {
        vector<vector<int>> vvi;
        // 排序
        sort(nums.begin(), nums.end());
        int n = nums.size();
        for(int i = 0; i < n;){
            // 两数之和
            int l = i + 1, r = n - 1;
            while(l < r){
                int sum = nums[l] + nums[r];
                int target = -nums[i];
                if(sum > target) r--;
                else if(sum < target) l++;
                else{
                    vvi.push_back({nums[i], nums[l], nums[r]});
                    l++;
                    r--;
                    //去重
                    while(l < r && nums[l] == nums[l - 1]) l++;
                    while(l < r && nums[r] == nums[r + 1]) r--;
                } 
            }
            // 去重
            i++;
            while(i < n && nums[i] == nums[i - 1]) i++;
        }
        return vvi;
    }
};

4. 知识与收获

( 1 ) (1) (1) 三数之和是两数之和的变形,理解了两数之和的核心思想,三数之和也就能够顺理成章的解决。
( 2 ) (2) (2) 理解为什么需要去重处理:
对于两数之和(a+b=target):一个组合(a, b)满足了条件,数组又是有序的,对于a来说,以后的所有与其相同的数如果也满足了条件,那么一定会与(a,b)组合相同;对于b来说与a同理。
对于三数之和(a+b+c=target):变形为(a+b=target-c),target确定,本次循环内c也确定


T h e The The E n d End End

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/437202.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

代码第二十四天-寻找旋转排序数组中的最小值Ⅱ

寻找旋转排序数组中的最小值Ⅱ 题目要求 解题思路 二分法 当遇到两个left、right两个位置值相同时候&#xff0c;可以选择将 right right-1 代码 class Solution:def findMin(self, nums: List[int]) -> int:left,right0,len(nums)-1while left<right:mid(leftright…

Python编程作业五:面向对象编程

目录 一、类的定义和方法 二、图书管理系统 一、类的定义和方法 定义一个学生类&#xff08;Student&#xff09;&#xff0c;包括学号(id)、姓名(name)、出生日期(birthday)和分数(score)4个属性&#xff0c;其中出生日期是私有属性&#xff0c;不能被外界直接访问。该类应具…

华容道问题求解_详细设计(三)之查找算法1_DFS

&#xff08;续上篇&#xff09; 使用DFS查找算法的原因是因为它符合本人的思考习惯&#xff0c;另外在第一版时用的就是这个方法&#xff0c;后来知道这不是查找这类问题的最好方法。 在前面的概要设计中的框图里描述的方法就是DFS&#xff0c;它可以找到一个解法&#xff0c;…

某省内存取证真题详解

需要环境的私我 题目描述: 一,从内存中获取到用户admin的密码并且破解密码,以Flag{admin,password}形式提交(密码为6位) 二,获取当前系统ip地址及主机名,以Flag{ip:主机名}形式提交; 三,当前系统中存在的挖矿进程,请获取指向的矿池地址,以Flag{ip}形式提交 四,恶意进…

大数据冷热分离方案

数据冷热分离方案 1、背景 ​ 随着业务的发展&#xff0c;在线表中的数据会逐渐增加。常规业务都有冷热数据现象明显的特性&#xff08;需要访问的都是近期产生的热数据&#xff1b;时间久远的冷数据出于备份、备案溯源等诉求会进行在线保留&#xff09;。在业务表数据 量可控…

问题总结,web自动化测试元素无法操作?shadowDOM节点元素解决......

前言 web自动化遇到shadowDOM你会操作吗&#xff1f; 之前在做web自动化的时候&#xff0c;发现页面上有些元素&#xff0c;在selenium中无法通过xpath来定位&#xff0c;各种原因找了半天都没找到解决方案&#xff0c;最后发现元素在一个叫做shadow-root的节点下面&#xff…

艺术与科技的结合,AI绘画图生图怎么样?

AI绘画图生图是指通过人工智能技术生成的具有艺术价值的图像。它可以根据用户提供的参考图像或描述&#xff0c;自动生成具有艺术风格的新图像。这些图像可以是风景、人物、抽象画等各种形式。那么ai绘画图生图到底怎么样&#xff1f; AI绘画图生图的优点在于它可以快速、高效地…

R语言安装IDE工具,RStudio 安装

R语言安装IDE工具&#xff0c;RStudio 安装 介绍下载安装包安装使用运行结果快捷键和使用技巧常用快捷键使用技巧 介绍 RStudio是一个集成开发环境&#xff08;IDE&#xff09;&#xff0c;专门用于R编程语言的开发和数据分析。它提供了许多工具和功能&#xff0c;使R编程更加…

初始网络 --- 网络基础

目录 0、 前言 1、 计算机网络发展背景 1.1. 局域网(LAN) && 广域网(WAN) 2、 认识并理解协议 3、 初始网络协议 3.1. 协议分层 4、 TCP/IP 五层(或四层)模型 4.1. 简单了解TCP/IP层状体系 4.2. TCP/IP协议层状结构和计算机层状结构的关系 5、 OSI七层模型 …

七.AV Foundation 视频播放 - 图片进度条

引言 播放器的功能功能已经十分完善了&#xff0c;接下来我们给它添加一些提升用户体验的功能。当前市面上的主流播放器几乎都有一个非常友善的功能&#xff0c;用户在退拽进度条的时候可以看见进度条所处进度的视频画面&#xff0c;这对于用户来说是一种直观而且便捷的体验。…

noetic ros配置因时机械夹爪的驱动

noetic ros配置因时机械夹爪的驱动文件 配置编译教程解决方案 配置编译教程 1.inspire_robot 包支持因时机器人公司的机械夹爪在ROS平台上的使用&#xff0c;我们在ros noetic环境下进行了测试。 2.为了使程序能够正常运行&#xff0c;需要执行以下环境配置操作&#xff1a;&a…

php:下拉列表查询(静态数据+数据库数据)

一、在php中嵌套 效果 1、从php中嵌套html语句 下拉列表的显示 echo <div class"text-nav-1 required "><div> . _(在职状态) . :</div> <select name"work_status">; // 定义选项数组 $options [all > _(全部),inwork &g…

越南电力展|2024年第17届越南国际电力设备与技术展览会

2024年第17届越南国际电力设备与技术展览会 The 17th International Exhibition on ELECTRICAL TECHNOLOGY & EQUIPMENT VIETNAM ETE 2023 同期举办&#xff1a;2024 年第 14 届越南节能和绿色能源科技产品博览会 The 14th International Exhibition on PRODUCTS TECHNO…

C语言--- qsort函数

目录 一.qsort函数 1.qsort函数的功能 2.四个参数讲解 (1)base (2)num (3)size (4)compare 3.使用qsort函数对一个整形数组进行排序 4.qsort函数排序结构体数据 第一种&#xff1a;按照年龄进行比较 第二种&#xff1a;按照名字进行排序 二.利用冒泡排序模仿qsort函…

慢sql优化记录1

慢sql为&#xff1a; select count(*) from t_wf_process p left join t_wf_core_dofile dofile on p.wf_instance_uid dofile.instanceid join zwkj_department d on p.userdeptid d.department_guid ,t_wf_core_item i,wf_node n where (p.IS_DUPLICATE ! true or p.IS_DU…

Leetcoder Day39| 动态规划part06 完全背包问题

完全背包理论 有N件物品和一个最多能背重量为W的背包。第i件物品的重量是weight[i]&#xff0c;得到的价值是value[i] 。每件物品都有无限个&#xff08;也就是可以放入背包多次&#xff09;&#xff0c;求解将哪些物品装入背包里物品价值总和最大。 示例&#xff1a; 背包最大…

2023第二届陇剑杯网络安全大赛 SS Writeup

sevrer save_1 打开流量包文件过滤http流量 从这个/helloworld/greeting开始追踪TCP流 直接百度搜索payload 搜索得到这题flag就是CVE-2022-22965 sevrer save_2 追踪TCP流&#xff0c;在tcp.stream eq 106&#xff0c;发现反弹shell的IP和端口 这题flag为192.168.43.128:2333…

PPT模板一键下载,原创精美,2024必备!

1. PPT模板分享 &#xff08;1&#xff09;计算机学院毕业答辩PPT &#xff08;2&#xff09;开学典礼活动策划方案PPT &#xff08;3&#xff09;新员工入职培训PPT &#xff08;4&#xff09;宠物行业分析报告PPT &#xff08;5&#xff09;机关青年干部述职PPT 以上PPT模板均…

centos离线安装 k8s (实操可用)

全部安装包rpm下载&#xff08;已整理好k8s和docker&#xff09;&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1ATv8BPijhvIKWz4hMnkx6Q?pwdt5db 提取码&#xff1a;t5db 将文件下载以后&#xff0c;解压到服务器 #执行所有docker-rpm包 yum -y localinstall *.rpm…

testvue-个人中心

header.vue(右上角) <template><div class="header"><!-- 折叠按钮 --><div class="collapse-btn" @click="collapseChage"><i v-if="!collapse" class="el-icon-s-fold"></i><…