LeetCode - 双指针(Two Pointers) 算法集合 [对撞指针、快慢指针、滑动窗口、双链遍历]

欢迎关注我的CSDN:https://spike.blog.csdn.net/
本文地址:https://spike.blog.csdn.net/article/details/139270999

Two Sum

双指针算法是一种常见且灵活的技巧,通过使用两个指针协同完成任务。这些指针可以指向不同的元素,具体应用取决于问题的性质。双指针算法的常见用法:

  1. 对撞指针:一左一右向中间逼近。例如,反转字符串中的元音字母问题,可以使用对撞指针。
  2. 快慢指针:一快一慢,步长不同。例如,判断链表中是否有环问题,可以使用快慢指针,看慢指针是否能追上快指针。单链表找中间节点问题也可以用快慢指针,快指针到链表结尾,慢指针到一半。
  3. 滑动窗口:类似计算机网络中的滑动窗口。一般是右端向右扩充,达到停止条件后右端不动,左端向右端逼近,逼近达到停止条件后,左端不动,右端继续扩充。

双指针算法包括:对撞指针快慢指针滑动窗口双链遍历

  1. 167. 两数之和 II - 输入有序数组 - 对撞指针
  2. 633. 平方数之和 - 对撞指针,题目167变种
  3. 680. 验证回文串 II - 对撞指针,需要判断去除1个字符 l+1 或 r-1
  4. 15. 三数之和 - 两数之和的进阶版,先排序,逐步修改序列
  5. 142. 环形链表 II - 快慢指针
  6. 76. 最小覆盖子串 - 滑动窗口,Counter类的使用
  7. 88. 合并两个有序数组 - 双链遍历
  8. 524. 通过删除字母匹配到字典里最长单词 - 双链遍历,优先排序,再依次比较。

1. 对撞指针

167. 两数之和 II - 输入有序数组 - 对撞指针

class Solution:def twoSum(self, numbers: List[int], target: int) -> List[int]:"""时间复杂度O(n),空间复杂度O(1)"""# 输入的已经排序n=len(numbers)  # 数量l,r=0,n-1  # 左右指针while l<r:v=numbers[l]+numbers[r]  # 两数之和if v>target: # 移动指针r-=1elif v<target:l+=1else:return [l+1,r+1]

633. 平方数之和 - 对撞指针,167变种

class Solution:def judgeSquareSum(self, c: int) -> bool:"""时间复杂度 O(n),空间复杂度 O(1)"""# 左右指针l, r = 0, int(sqrt(c))while l <= r:  # 遍历条件需要相等# 计算值v = pow(l, 2) + pow(r, 2)if v > c:  # 移动右指针r -= 1elif v < c:  # 移动左指针l += 1else:return Truereturn False

680. 验证回文串 II - 对撞指针,需要判断去除1个字符 l+1 或 r-1

class Solution:def validPalindrome(self, s: str) -> bool:"""时间复杂度 O(n), 空间复杂度 O(1)"""def check(l, r):"""检查s是否是回文"""while l < r:if s[l] == s[r]:l += 1r -= 1else:return Falsereturn Truen = len(s)l, r = 0, n-1  # 左右指针while l < r:if s[l] == s[r]:  # 回文l += 1r -= 1else:# 越过1个值,最多可以从中删除一个字符return check(l+1, r) or check(l, r-1)return True

15. 三数之和 - 两数之和的进阶版,先排序,逐步修改序列

class Solution:def threeSum(self, nums: List[int]) -> List[List[int]]:"""时间O(N^2),空间O(logN) -> 排序"""def two_sum(nums, t):"""经典的两数之和,同时,避免重复添加"""n = len(nums)l, r = 0, n-1res = []while l < r:x = nums[l] + nums[r]if x > t:r -= 1elif x < t:l += 1else:sr = [nums[l], nums[r], -t]if sr not in res:  # 避免重复添加res.append(sr)l += 1return resnums.sort()  # 排序n = len(nums)  # 序列长度res = []  # 输出结果for i in range(n):# 避免重复数字if i > 0 and nums[i-1] == nums[i]:continuet = nums[i]  # 依次遍历res += two_sum(nums[i+1:], -t)return res

2. 快慢指针

142. 环形链表 II - 快慢指针

# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = Noneclass Solution:def detectCycle(self, head: Optional[ListNode]) -> Optional[ListNode]:"""时间复杂度O(n),空间复杂度O(1)"""if not head:return None# 快慢指针先指向headslow=fast=head# 判断是否运行到结尾while fast.next and fast.next.next:slow=slow.next  # 移动1步fast=fast.next.next  # 移动2步if slow==fast:fast=head # fast从head开始重新计数while slow!=fast:  # 移动到位置slow=slow.nextfast=fast.nextreturn fast # 相等即返回return None

3. 滑动窗口

76. 最小覆盖子串 - 滑动窗口

class Solution:def minWindow(self, s: str, t: str) -> str:"""时间复杂度 O(m+n),空间复杂度 O(1)"""rl,rr=-1,len(s) # 目标的最大窗口l=0  # 左指针cnt_s=Counter()  # 计数器字典cnt_t=Counter(t) # 目标计数器字典less=len(cnt_t)  # 不重复的t字母数量for r,c in enumerate(s):cnt_s[c]+=1  # s计数器if cnt_s[c]==cnt_t[c]:  # 数量相等less-=1  # 数量减一while less==0:  # 满足条件if r-l < rr-rl:  # 区间更小rl,rr=l,r  # 更新左右指针# -----# 如果数量相等,之后数量需要减一,所以less需要提前+1x=s[l]  # 左指针当前字母if cnt_s[x]==cnt_t[x]:  less+=1  # 不重复+1cnt_s[x]-=1  # 指针减1l+=1  # 已经更新,所以需要移动左指针# -----return "" if rl<0 else s[rl:rr+1]

4. 双链遍历

88. 合并两个有序数组 - 双链指针

class Solution:def merge(self, nums1: List[int], m: int, nums2: List[int], n: int) -> None:"""Do not return anything, modify nums1 in-place instead.时间复杂度 O(n),空间复杂度 O(1)"""pos=(m-1)+(n-1)+1  # 最远位置m-=1  # 最后位置n-=1  # 最后位置while m>=0 and n>=0:  # 遍历两个数组# 注意: 大于等于确保,优先移动mif nums1[m]>=nums2[n]:nums1[pos]=nums1[m]  # 赋值大值m-=1else:nums1[pos]=nums2[n]n-=1pos-=1  # 移动指针while n>=0: # 优先移动m,所以剩下的就是nnums1[pos]=nums2[n]pos-=1n-=1

524. 通过删除字母匹配到字典里最长单词 - 双链指针,优先排序,再依次比较。

class Solution:def findLongestWord(self, s: str, dictionary: List[str]) -> str:"""d 是字典长度,m 是s长度时间复杂度 O(d x (m+n)),空间复杂度 O(d x m)"""# 按目标结果排序,优先做前面的words = sorted(dictionary, key=lambda x: (-len(x), x))for t in words:i = j = 0  # 双指针while i < len(t) and j < len(s): # 双指针遍历if t[i] == s[j]:  # 相同i += 1  # t指针+1,满足条件j+=1  # s指针默认都+1if i == len(t):  # 长度满足return t  # 直接返回return ""

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是勒索软件

什么是勒索软件 勒索软件又称勒索病毒&#xff0c;是一种特殊的恶意软件&#xff0c;又被归类为“阻断访问式攻击”&#xff08;denial-of-access attack&#xff09;&#xff0c;与其他病毒最大的不同在于攻击手法以及中毒方式。勒索软件的攻击方式是将受害者的电脑锁起来或者…

Spring Boot中如何查询PGSQL分表后的数据

数据库用的pgsql&#xff0c;在表数据超过100w条的时候执行定时任务进行了分表&#xff0c;分表后表名命名为原的表名后面拼接时间&#xff0c;如原表名是card_device_trajectory_info&#xff0c;分表后拼接时间后得到card_device_trajectory_info_20240503&#xff0c;然后分…

ubuntu使用oh my zsh美化终端

ubuntu使用oh my zsh美化终端 文章目录 ubuntu使用oh my zsh美化终端1. 安装zsh和oh my zsh2. 修改zsh主题3. 安装zsh插件4. 将.bashrc移植到.zshrcReference 1. 安装zsh和oh my zsh 首先安装zsh sudo apt install zsh然后查看本地有哪些shell可以使用 cat /etc/shells 将默…

使用nexus搭建的nodejs私库,定期清理无用的npm组件,彻底释放磁盘空间

一、背景 昨天我们整理了一篇关于docker私库&#xff0c;如何定期清理以释放磁盘空间的文章。 虽然也提及了npm前端应用的组件该如何定期清理的&#xff0c;本文是对它作一个补充说明。 前文也看到了&#xff0c;npm组件占用的blob空间为180多GB&#xff0c;急需清理。 二、…

100个 Unity小游戏系列三 -Unity 抽奖游戏专题一 转盘抽奖游戏

一 、效果展示 二、知识点 2.1 布局需要实现功能 1、转动的根目录为itemSpinRoot 2、创建对应的item 3、每个item转动的角度 2.2 代码 public class WheelDialog : UIBase{[SerializeField] Button btnClick;[SerializeField] Button btnClose;[SerializeField] Sprite[] ite…

哪有异地组网的工具?

不同地区的电脑与电脑、设备与设备、电脑与设备之间的信息远程通信&#xff0c;一直是企业和个人面临的难题。通过使用天联组网的解决方案&#xff0c;这个问题将迎刃而解。 天联组网解决方案 天联组网是一种可以实现不同地区之间电脑、设备及其之间的信息远程通信的解决方案。…

Trie字符串统计-java

Trie&#xff0c;又称前缀树或字典树&#xff0c;是一种有序树&#xff0c;用于保存关联数组&#xff0c;其中的键通常是字符串。 目录 前言☀ 一、Trie字符串统计☀ 二、算法思路☀ 1.Trie树定义&#x1f319; 2.变量解释&#x1f319; 3.插入操作&#x1f319; 4.Trie树查找操…

【class18】人工智能初步----语音识别(4)

【class17】 上节课&#xff0c;我们学习了: 语音端点检测的相关概念&#xff0c;并通过代码切分和保存了音频。 本节课&#xff0c;我们将学习这些知识点&#xff1a;1. 序列到序列模型2. 循环神经网络3. 调用短语音识别接口 知其然&#xff0c;知其所以然 在调用语…

数组单调栈-901. 股票价格跨度、leetcode

单调栈作为一种数据结构在求解类递增、递减方面的题目中有较为广泛的应用&#xff0c;在以往的leetcode中所见到的相关单调栈的题目均为单一元素&#xff0c;今天刷到901题目时&#xff0c;想到了将数组元素作为单调栈中元素的方法进行求解。 题目链接及描述 901. 股票价格跨…

【c++leetcode】69. Sqrt(x)

问题入口 二分搜索 最困难的是能否意识到用二分搜索法解题。 算术平方根的区间在[1, x] 。代码如下&#xff1a; class Solution { public:int mySqrt(int x) {if (x 1 || x 0){return x;}int64_t start 1;int64_t end x;while (start < x){int64_t mid start (en…

以果决其行,只为文化的传承

从他们每一个人的身上&#xff0c;我们看到传神的东西&#xff0c;就是他们都能用结果&#xff0c;去指引自己前进的方向&#xff0c;这正是我要解读倪海厦老师的原因&#xff0c;看倪海厦2012年已经去世&#xff0c;到现在已经十几年时间了&#xff0c;但是我们看现在自学中医…

【Pandas】深入解析`pd.to_sql()`函数

【Pandas】深入解析pd.to_sql()函数 &#x1f308; 欢迎莅临我的个人主页&#x1f448;这里是我深耕Python编程、机器学习和自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;并乐于分享知识与经验的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1…

2024年第六届中青杯数学建模竞赛浅析

获取比赛资料&#xff0c;请关注gzh“小何数模”&#xff01; 本次中青杯数学建模的赛题已正式出炉&#xff0c;无论是赛题难度还是认可度&#xff0c;该比赛都是仅次于数模国赛的独一档&#xff0c;可以用于国赛前的练手训练。考虑到大家解题实属不易&#xff0c;为了帮助大家…

JavaSE:StringBuilder和StringBuffer类

1、引言 在上一篇文章中&#xff0c;我们理解了字符串的常用方法&#xff0c;细心的同学大概已经发现&#xff0c;不管是将字符串中的字符转变为大写或小写&#xff0c;或是完成字符串的替换&#xff0c;又或是去除空白字符等等&#xff0c;只要涉及到字符串的修改&#xff0c…

【PB案例学习笔记】-10 进度条使用

写在前面 这是PB案例学习笔记系列文章的第10篇&#xff0c;该系列文章适合具有一定PB基础的读者。 通过一个个由浅入深的编程实战案例学习&#xff0c;提高编程技巧&#xff0c;以保证小伙伴们能应付公司的各种开发需求。 文章中设计到的源码&#xff0c;小凡都上传到了gite…

Java用反射reflect来实例化对象: class.getDeclaredConstructor().newInstance()

Java用反射reflect来实例化对象: class.getDeclaredConstructor().newInstance() 从java9开始, class.newInstance()已过时, 被加上Deprecated强烈反对注解 SuppressWarnings("removal")CallerSensitiveDeprecated(since"9")public T newInstance()throws …

防止自动化攻击的最佳实践

防止自动化攻击的最佳实践 在当今的网络安全环境中&#xff0c;保护用户账户免受自动化攻击已成为每个网站和应用程序的重要任务。攻击者可以利用多种不同类型的自动化攻击来尝试破坏用户账户。本文将详细介绍常见的攻击类型及其防御机制&#xff0c;帮助您更好地保护用户账户…

adb 连接机顶盒命令

抓机顶盒日志的方法&#xff0c;使用此命令进行抓日志&#xff0c;个别无法抓日志的盒子可以使用此方法 1、安卓9.0版本查询命令 ps -ef |grep com.cm.webos.iptv 2、安卓4.4版本查询命令 ps |grep com.cm.webos.iptv 3、查询顺序&#xff1a;首先进入shell下进行操作 adb she…

C++青少年简明教程:for循环语句

C青少年简明教程&#xff1a;for循环语句 C的for循环语句是一种迭代控制语句&#xff0c;用于重复执行一段代码。 语法格式&#xff1a; for(表达式1&#xff1b;表达式2&#xff1b;表达式3) 循环体 for循环语句执行流程图&#xff1a; 不太好理解&#xff0c;请看下图&am…

VSCode配置Lua5.4安装

参考&#xff1a;VSCode 配置 Lua 开发环境(清晰明了)_lua vscode-CSDN博客 1.下载 Lua Binaries Download (sourceforge.net) 2.配置环境变量 解压放到某文件夹&#xff1a; 环境变量&#xff1a; 3.VSCode安装插件 4.配置 5.测试