【LeetCode 算法专题突破】双指针(⭐)
文章目录
- 前言
- 1. 移动零
- 题目描述
- 代码
- 2. 复写零
- 题目描述
- 代码
- 3. 快乐数
- 题目描述
- 代码
- 4. 盛最多水的容器
- 题目描述
- 代码
- 5. 有效三角形的个数
- 题目描述
- 代码
- 6. 三数之和
- 题目描述
- 代码
- 7. 四数之和
- 题目描述
- 代码
- 总结
前言
学算法入门必学的一个章节,双指针算法,不说废话,直接开始。
1. 移动零
我们先来一道经典的双指针题目试试水
题目链接:283. 移动零
题目描述
怎么样才能在不创建新数组的情况下把 0 移动到数组的末尾呢?(如果不是有这个要求我肯定也无脑创建一个数组遍历解决)来看代码:
代码
func moveZeroes(nums []int) {slow, fast := 0, 0for fast < len(nums) {if nums[fast] != 0 {tmp := nums[fast]nums[fast] = nums[slow]nums[slow] = tmpslow++}fast++}
}
我们设置 slow,fast 两个指针,都从 0 开始遍历,fast 不断向后遍历查找不等于 0 的数交给 slow 位置,这样就会出现一种情况:
以 slow 为分界线,左边的区间都是排好的数字,右边就是没排好的数字,最后左边就排好了,而右边就都剩下 0 了
形象点说就是 fast 把数字都按序丢到了左边的区间
2. 复写零
像这样需要在数组中移动、删除、增加元素这类的操作,都是离不开双指针的算法思想
题目链接:1089. 复写零
题目描述
如果没有要求原地解决这道题目,也会非常的简单,但是如果需要我们原地求解,又好像有点无从下手了(如果使用 insert 方法来做这道题,那复杂度会非常的高)来看代码如何解决:
代码
func duplicateZeros(arr []int) {left, right := 0, 0// 找到一共经过了几个 0, 调整好位置for right < len(arr) {if arr[left] == 0 { // 注意这里用的是 leftright++}left++right++}left--right--// 反向遍历for left >= 0 {if right < len(arr) {arr[right] = arr[left]}if arr[left] == 0 { // 复写 0 的操作right--arr[right] = 0}left--right--}
}
这道题虽然是简单题,但是想要按照题目的要求来做出这道题并不容易,我们如果是第一次做,是很难想到使用反向迭代的双指针来做,所以还是重复我的一个观点:多刷算法,多积累,见多识广才能思路开阔。至少下一次遇到类似的题目我们就能匹配一下这个思路了。
回到正题,这道题如果使用双指针的话,需要我们从后往前迭代使用,直接用嘴说可能不太好理解,我给出的建议是,用题目给出的示例,然后对着代码把流程跑一遍,这样你能了解到这个算法的基本思路,可以看看图解:https://blog.csdn.net/Locky136/article/details/131537158
这里有两个需要注意的点:
- 一开始根据 left 经过多少个零来调整两个指针的位置(这里我用 left 和 right 命名是为了分辨两个指针的相对位置)
- 反向遍历时复写零的操作(为什么只复写了一次?因为还有一次和上一个操作合并了)
3. 快乐数
双指针还有一个非常重要的思想,就是快慢指针的思想,其实听名字大家差不多都能猜到具体的用法,但我们还是得来一道经典的题目试试水
题目链接:202. 快乐数
题目描述
这道题题意很好理解,所以我们直接根据示例,不说废话,直接上图:(题目的示例二,我们来模拟他的流程)
我们可以看到就像一个环,用快慢指针遍历,如果快指针追上了慢指针,那不就代表这段代码无限循环了吗?来看代码:
代码
func isHappy(n int) bool {Sum := func(n int) int { // 进行一次快乐数的计算sum := 0for n > 0 {tmp := n%10sum += tmp*tmpn /= 10}return sum}fast, slow := n, nfor {slow = Sum(slow)fast = Sum(Sum(fast))if fast == slow {break;}}return fast == 1
}
其实这段算法的核心思想前面已经解释的很清楚了,这里就不赘述了,如果还有疑问跟着上面的图片走一遍代码就行~
4. 盛最多水的容器
再来一道经典的双指针题目练练手,如果没刷过这道题,你怎么敢说你刷过 LeetCode 呢~
题目链接:11. 盛最多水的容器
题目描述
这道题的思路说难不难,说简单其实也不一定想的出来(首先把暴力排除,用暴力匹配就没意思了,更何况 10^5 的数据量用 O(N^2) 的算法不一定能过完这些样例,可能会超时(不然也不是中等难度的题目了))那我们就来看看怎么用双指针来做出这道题吧~
代码
func maxArea(height []int) int {left, right, max := 0, len(height)-1, 0for left < right {tmp := Min(height[left], height[right])*(right-left) // 计算当前容量max = Max(max, tmp) // 迭代出最大容量if height[left] > height[right] {right--} else {left++}}return max
}func Min(a, b int) int {if a > b {return b}return a
}func Max(a, b int) int {if a > b {return a}return b
}
在解释思路之前必须吐槽一下 LeetCode 的 Golang 编译器,最新版本的 Go 其实已经实装了 max 和 min 让我们更方便的求最大值和最小值,但是我试了一下,LeetCode 的编译器并没有支持这个版本,搞得我只好自己写找最大最小值的方法,可恶
这道题其实知道想清楚核心的思路就很好做,left 和 right 指针分别在两边,我们只需要让比较矮的那一边的柱子移动即可,因为如果让较高的柱子移动,容量只可能更小,而让较矮的柱子移动:如果遇到更高的柱子,容量就可能更大。(这感觉也有一点贪心的思想在里面),根据分析,我们只要不断移动较矮的柱子,就能求出最大的容量了。
5. 有效三角形的个数
咱们再来做一道题目,练习双指针~
题目链接:611. 有效三角形的个数
题目描述
听完题意,有点算法经验的我们都能看出这道题目,如果使用暴力来做,那复杂度会非常的高,而且也并不好写,那我们该怎么设计算法来解决这道题目?来看代码:
代码
func triangleNumber(nums []int) int {sort.Ints(nums)ans := 0for i := len(nums)-1; i >= 0 ; i-- {left, right := 0, i-1for left < right {if (nums[left]+nums[right]) > nums[i] {ans += (right-left)right-- } else {left++}}}return ans
}
这段代码的核心流程就是,将数组中最大的数作为三角形最长的那条边,也就是 i,利用三角形两边之长大于第三边的性质,快速找出符合要求的三角形,通过使用双指针的形式遍历余下的两条三角形的边
如果听完,代码也模拟完之后还有疑问的话,可以去看这篇文章,我曾经写过的详细文字解答,还配有清晰的图解:https://blog.csdn.net/Locky136/article/details/131590581
6. 三数之和
刷了这么多双支指针的题目了,怎么能错过 LeetCode 中大名鼎鼎的几数之和系列呢,两数之和,LeetCode 的第一道题,那是多少人刷题的起点,和梦想的开始呀,但是我们肯定不能还去做这么简单的题目,那必须是从三数之和开始刷起
题目链接:15. 三数之和
题目描述
这道题目其实并没有什么高深的技巧,抑或是复杂的算法,更多的是考察我们对代码的掌控能力,看看我们能不能写出一个像样的双指针算法,来看代码:
代码
func threeSum(nums []int) [][]int {sort.Ints(nums)var ans [][]intn := len(nums)-1for i, v := range nums[:n-1] {if i != 0 && v == nums[i-1] { // 跳过重复数字continue}if v+nums[i+1]+nums[i+2] > 0 { // 三数相加无论如何都会 > 0, 不需要再遍历 break}if v+nums[n]+nums[n-1] < 0 { // 三数最大值 < 0, 让 i 继续遍历continue}// 双指针left, right := i+1, nfor left < right {s := v+nums[left]+nums[right]if s > 0 {right--} else if s < 0 {left++} else {ans = append(ans, []int{v, nums[left], nums[right]})// 跳过重复数字for left < right && nums[left] == nums[left+1] {left++}for left < right && nums[right] == nums[right-1] {right--}left++right--}}}return ans
}
思路其实很简单,依旧是遍历一个 i,作为起点,用双指针匹配出符合题目要求的值,最后拼接在一起返回即可,这里需要注意的是我们需要根据题目的要求跳过重复的数字,不然最后还得去重,那反而就更麻烦了。
7. 四数之和
接下来就是我们练习双指针的最后一道题目啦,四数之和~
题目链接:https://leetcode.cn/problems/4sum/
题目描述
四数之和相较于三数之和,需要枚举的数字多了一个,我们只需要在三数之和的条件下,多枚举一个数即可,不过对代码掌控能力的要求也随之升高了不少,来看代码:
代码
func fourSum(nums []int, target int) [][]int {sort.Ints(nums)var ans [][]intn := len(nums)-1for a := 0; a < n-2; a++ {value1 := nums[a]if a != 0 && value1 == nums[a-1] { // 跳过重复数字continue}if value1+nums[a+1]+nums[a+2]+nums[a+3] > target { // 四数之和 > target break}if value1+nums[n-2]+nums[n-1]+nums[n] < target { // 四数最大和 < targetcontinue}for b := a+1; b < n-1; b++ {value2 := nums[b]if b != a+1 && value2 == nums[b-1] { // 跳过重复数字continue}if value1+value2+nums[b+1]+nums[b+2] > target { // 四数之和 > target break}if value1+value2+nums[n-1]+nums[n] < target { // 四数最大和 < targetcontinue}left, right := b+1, nfor left < right {sum := value1+value2+nums[left]+nums[right]if sum > target {right--} else if sum < target {left++} else {ans = append(ans, []int{value1, value2, nums[left], nums[right]})// 跳过重复数字for left < right && nums[left] == nums[left+1] {left++}for left < right && nums[right] == nums[right-1] {right--}left++right--}}}}return ans
}
省流:其实跟三数之和换汤不换药,就是多加上了一层循环,仅此而已,代码一下就能看的出来。不过为了更好的掌控代码,我就没有再用 Golang 提供的语法糖,而是老老实实的用 for 循环了。
总结
我刷过的,个人喜欢的,比较经典的,数组相关的双指针问题大概就是这些了,其实双指针并不能算是一个标准的算法,我个人更倾向于这是一个思考的方向,一个算法的基本素养。为什么这么说呢?因为之后我们去做更多其他算法题,多多少少都会用到这样一个思想,在做链表专题的时候,也会用到双指针的思想
所以想要学好算法,打好每一步的基础都是非常重要滴~
相关文章:
【LeetCode 算法专题突破】双指针(⭐)
文章目录 前言1. 移动零题目描述代码 2. 复写零题目描述代码 3. 快乐数题目描述代码 4. 盛最多水的容器题目描述代码 5. 有效三角形的个数题目描述代码 6. 三数之和题目描述代码 7. 四数之和题目描述代码 总结 前言 学算法入门必学的一个章节,双指针算法࿰…...
ts知识点——基础积累
第一章 快速入门 1、TypeScript简介 TypeScript是JavaScript的超集。它对JS进行了扩展,向JS中引入了类型的概念,并添加了许多新的特性。TS代码需要通过编译器编译为JS,然后再交由JS解析器执行。TS完全兼容JS,换言之,…...
mybatis plus MetaObjectHandler 不生效
首先要知道,spring boot 只会加载启动类同级和下级的bean 如果把bean放在启动类不同的上级目录,是加载不了bean的 如果把mybatisplus的配置文件放在与启动类不同包,就会扫描不到 例如放在这里,就扫描不到 放到这里,就可以扫描到...
力扣第216 组合总和 ||| c++ 回溯 + 注释
题目 216. 组合总和 III 中等 相关标签 数组 回溯 找出所有相加之和为 n 的 k 个数的组合,且满足下列条件: 只使用数字1到9每个数字 最多使用一次 返回 所有可能的有效组合的列表 。该列表不能包含相同的组合两次,组合可以以任何顺…...
深度学习系列51:hugging face加速库optimum
1. 普通模型 Optimum是huggingface transformers库的一个扩展包,用来提升模型在指定硬件上的训练和推理性能。Optimum支持多种硬件,不同硬件下的安卓方式如下: 如果是国内安装的话,记得加上-i https://pypi.tuna.tsinghua.edu.c…...
【QT开发笔记-基础篇】| 第四章 事件QEvent | 4.6 定时器事件
本章要实现的整体效果如下: QT 中使用定时器,有两种方式: 定时器类:QTimer定时器事件:QEvent::Timer,对应的子类是 QTimerEvent 本节通过一个案例,同时讲解这两种方式 案例:当点击…...
阿里云服务器ECS实例规格族c/g/r等字母说明
阿里云服务器ECS实例命名规则:ecs.<规格族>.large字母含义命名说明,包括x86、ARM架构、GPU异构计算、弹性裸金属、超级计算集群SCC云服务器,c代表计算型、g代表通用型、r代表内存型、u代表通用算力型、e代表经济型e实例,阿里…...
Everything和SVN结合使用-在Everything中显示SVN
点击跳转>Unity3D特效百例点击跳转>案例项目实战源码点击跳转>游戏脚本-辅助自动化点击跳转>Android控件全解手册点击跳转>Scratch编程案例点击跳转>软考全系列 👉关于作者 专注于Android/Unity和各种游戏开发技巧,以及各种资源分享&…...
代码随想录算法训练营第五十二天| 123.买卖股票的最佳时机III 188.买卖股票的最佳时机IV
今日学习的文章链接和视频链接 123.买卖股票的最佳时机III 视频讲解:https://www.bilibili.com/video/BV1WG411K7AR https://programmercarl.com/0123.%E4%B9%B0%E5%8D%96%E8%82%A1%E7%A5%A8%E7%9A%84%E6%9C%80%E4%BD%B3%E6%97%B6%E6%9C%BAIII.html 188.买卖股票的…...
②. GPT错误:图片尺寸写入excel权限错误
꧂问题最初 ꧁ input输入图片路径 print图片尺寸 大小 长宽高 有颜色占比>0.001的按照大小排序将打印信息存储excel表格文件名 表格路径 图片大小 尺寸 颜色类型 占比信息input输入的是文件就处理文件 是文件夹📁就处理文件。路径下的图片 1. 是处理本路径图片 …...
JQuery、JSON、AJAX、XML、IO流、多线程、反射核心知识点详解
JQuery 一、什么是JQuery JQuery是JavaScript的一个框架,对js的封装,使得js简单易学 优点: 1、不用考虑浏览器兼容性问题 2、jquery拥有强大的选择器,简化了js代码 3、jquery提供了很多系统函数,直接调用 二、版本 1.x…...
基于python的多种图像增强算法实现
基于python的多种图像增强算法实现 引言工具算法增强对比度直方图均衡化锐化图像噪声消除中值滤波均值滤波高斯滤波双边滤波增强对比度直方图均衡化总结全部资源引用引言 本项目使用python实现多种空域增强的图像增强算法,并使用了pyqt编写页面。通过点击不同页面的多种按钮,…...
Java前后端交互实现班级管理(查询)
1,数据库创建存储专业信息的表 2,后端: 连接数据库工具类DBUtil.java: package com.ffyc.webserver.util;import java.sql.*;public class DButils {static {try {Class.forName("com.mysql.cj.jdbc.Driver");} catch…...
论文速递 | 8月下旬9月上旬Operations ResearchManagement Science文章精选
编者按 本期我们选取了8月下旬及9月上旬Operations Research文章2篇,Management Science文章4篇期刊文章,着眼于各种不同场景下对于风险的预测、量化及管理,通过聚焦于风险这一主题,体系化地形成文章精选。 文章1 Computation of…...
DataBinding使用报错
val dataBinding DataBindingUtil.setContentView<ActivityMainBinding>(this,R.layout.activity_main)报错一: Unresolved reference: ActivityMainBinding 首先你要知道一个概念,ActivityMainBinding是DataBinding中的一种视频绑定ÿ…...
08Maven中的继承和聚合的作用
Maven中的继承 实际开发中对一个比较大型的项目进行了模块拆分 , 一个project下面创建了很多个modul, 每一个module都需要配置自己的依赖信息 开发中使用的同一个框架内的不同jar包,它们应该是同一个版本,所以整个项目中使用的框架版本需要统一 传统方…...
Ansible运行临时命令及常用模块介绍
目录 一.运行临时命令 1.基本语法格式 2.查看当前版本已安装的所有模块 二.ansible常见模块 1.command模块 2.shell模块 3.raw模块 4.script模块 5.file模块 参数列表: 示例: 6.copy模块 参数列表: 示例: 7.fetch模…...
EtherCAT报文-APRD(自动增量读)抓包分析
0.工具准备 1.EtherCAT主站 2.EtherCAT从站(本文使用步进电机驱动器) 3.Wireshark1.EtherCAT报文帧结构 EtherCAT使用标准的IEEE802.3 Ethernet帧结构,帧类型为0x88A4。EtherCAT数据包括2个字节的数据头和44-1498字节的数据。数据区由一个或…...
论文阅读:Seeing in Extra Darkness Using a Deep-Red Flash
论文阅读:Seeing in Extra Darkness Using a Deep-Red Flash 今天介绍的这篇文章是 2021 年 ICCV 的一篇 oral 文章,主要是为了解决极暗光下的成像问题,通过一个深红的闪光灯补光。实现了暗光下很好的成像效果,整篇文章基本没有任…...
将license验证加入到系统中
1.将ClientDemo下的cn文件夹的内容导入项目对应的java目录下。 2.将license-config.properties文件导入resources目录下。 3.在项目的pom.xml中添加如下依赖。 <properties><!-- Apache HttpClient --><httpclient>4.5.5</httpclient><!-- License…...
中断机制-interrupt和isInterrupted源码分析、中断协商案例
当前线程的中断标识为true,是不是线程就立刻停止? 答案是不立刻停止,具体来说,当对一个线程,调用interrupt时: 如果线程处于正常活动状态,那么会将该线程的中断标志设置为true,仅此…...
我与COSCon的故事【时光的故事】
曾经 2019年的时候,我还在日本读研究生,做一些物联网 (Internet of Things, IoT) 网络中的底层P2P (Peer to Peer) 通讯仿真模拟。这个方向是新来的Nguyen老师的新方向,它跟计算机强相关,但是很小众,实验室里也没有前辈…...
【科学文献计量】利用pybibx分析Scopus文献数据集(EDA,N-Grams,Cluster,Network analysis,NLP)
利用pybibx分析Scopus文献数据集 1 运行前准备1.1 数据集1.2 前置库2 加载库3 数据导入4 探索式数据分析,即EDA4.1 表格可视化4.2 词云图可视化4.3 N-Grams可视化4.4 文献聚类4.5 主题词演化4.6 桑基图可视化4.7 树图可视化4.8 作者生产力可视化5 网络可视化5.1 文献引用与被引…...
-带你看懂11种API类型及应用-
一起走进多样的API,多样的精彩 随着互联网行业的日益发展,API(Application Programming Interface)这个名词对于绝大多数来说都已不再陌生。然而,实际上,根据不同标准可以划分出不同类型的API。今天,让我们来走…...
集成友盟qq互联分享,导出风险问题处理
处理方案:移除 android:exported"true"即可。 注意友盟SDK QQ share 里默认配置是android:exported"true",所以要覆盖即可。...
探索数字安全的卓越之选 - Digicert证书
在数字时代,数据安全和隐私保护变得尤为重要。无论是个人网站、电子商务平台还是大型企业,保护用户数据和建立信任都是至关重要的任务。在这个领域,Digicert是一个备受推崇的品牌,提供了卓越的数字证书解决方案,以确保…...
第五章 流程控制 Pro
五、流程控制 1、条件语句 一、if语句(三种形式) 1、单分支语句: if (表达式)语句; //表达式可以是任何表达式 0和非0 多条语句加{ }构成复合语句 2、双分支语句 if(表达式) 语句1; else 语句2…...
CSS之实现线性渐变背景
1. background: linear-gradient() background: linear-gradient是CSS中用于创建线性渐变背景的属性,这个属性允许你定义一个在元素的背景中进行渐变的效果,可以从一个颜色过渡到另一个颜色。 基本语法 background: linear-gradient(direction, color-…...
软考 系统架构设计师系列知识点之特定领域软件体系结构DSSA(7)
接前一篇文章:软考 系统架构设计师系列知识点之特定领域软件体系结构DSSA(6) 所属章节: 第7章. 系统架构设计基础知识 第5节. 特定领域软件体系结构 相关试题 5. 特定领域软件架构(Domain Specific Software Archite…...
CentOS-7网卡重启后关闭的解决方法
第一步查找网卡: ip addr 如下图: 于是发现网卡eth0。 第二步进入网卡配置目录并进行配置: cd /etc/sysconfig/network-scriptsvim ifcfg-eth0 第三步改配置如下图: 然后每次重启后网卡会自动启动。...
购物网站开发公司/黑马培训机构可靠吗
转载于:https://www.cnblogs.com/joshuali/archive/2009/08/31/4339130.html...
国内包装设计网站/百度浏览器打开
第一次, 写点啥呢? 分享一下订阅的资源吧! 1 <?xml version"1.0" encoding"UTF-8"?>2 3 <opml version"1.0">4 <head>5 <title>建勇 subscriptions in feedly Cloud</title>6 </head>7…...
wordpress用户上传图片/搜索引擎推广的优势
在日常上网体验中,你肯定经常遇到类似的场景: 当你浏览短视频APP时,你会发现根本停不下来,因为相似的视频仿佛永远看不完。 当你正在浏览内容站点时,会发现看到一篇文章后,系统推荐的文章也是你所关心的话…...
企业网站手机端开发/seo搜索引擎工具
多态应用 应用实例 多态数组 数组的定义类型为父类类型,里面保存的实际元素类型为子类类型 现有一个继承结构如下:要求创建一个Person对象、2个student对象 和两个Teacher有一个teach,同一放在数组中,并调用say方法 publ…...
南京哪个网站做物业贷/百度知道电脑版网页入口
《本文转译自 Microsoft Security Response Center 博客文章“Update: MS10-015 security update re-released with new detection logic”》 大家好! 写这篇文章的目的是告诉大家,我们已经修订了 MS10-015 的安装包,使用了新的检测逻辑&…...
百度站长平台验证网站/今天的新闻主要内容
centos7 配置vlancentos7在网上找到的方法都是使用vconfig去配置,但偏偏centos7没有vconfig这个包,用yum安装也搜不到这个包,如果用centos6的包又太过麻烦,现提供已下配置方法注:此方法通过安装centos7时,配…...