JavaScript刷LeetCode拿offer-双指针技巧(上)

模式流星
• 阅读 684

一、前言

  一般情况下,遍历数组(或者字符串)操作,都是采用单指针从前往后或者从后往前依次访问数组(或者字符串)中的元素。

  而对于以下情况,只采用单指针处理,则会徒增时间复杂度和空间复杂度:

  • 例如:找到两个数使得它们相加之和等于目标数,采用单指针处理,则需要嵌套循环,使得时间复杂度增长为 O(n^2);
  • 再例如:翻转数组,采用单指针处理,则需要额外内存空间,使得空间复杂度增长为 O(n);

利用双指针技巧则可以优化上述解决方案:

  • 第一个例子:可以先对采用数组进行排序预处理,再创建前后指针向中间遍历,遍历的时间复杂度为 O(n),整体时间复杂度主要取决于排序算法,通常为 O(nlogn);
  • 第二个列子:一个指针负责遍历,另外一个指针负责交换元素,从而使得空间复杂度为 O(1);

双指针没有复杂的定义,总结起来主要处理两类问题:

  • 将嵌套循环转化为单循环问题
  • 通过指针记录状态,从而优化空间复杂度

下面的实战分析会让你感受双指针的威力。

二、167. 两数之和 II - 输入有序数组

给定一个已按照升序排列 的有序数组,找到两个数使得它们相加之和等于目标数。函数应该返回这两个下标值 index1 和 index2,其中 index1 必须小于 index2。

  这道题目采用单指针的做法只能通过嵌套循环枚举所有两数之和的方法来解决,时间复杂度为 O(n^2)。

  恰巧本题中的数组已经是有序数组,那么直接创建前后指针:

  • 如果两数之后大于 target,尾指针向前移动;
  • 如果两数之和小于 target,头指针向后移动;

JavaScript刷LeetCode拿offer-双指针技巧(上)

上述代码利用双指针技巧成功地将时间复杂度降低为 O(n)。

三、344. 反转字符串

编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 char[] 的形式给出。

  本题采用单指针的方法,需要创建一个额外的数组来保存翻转后的元素,空间复杂度为 O(n)。

  利用双指针技巧,则可以在遍历的过程中同时完成交换元素的操作,时间复杂度降低为 O(1)

JavaScript刷LeetCode拿offer-双指针技巧(上)

  相同类型的题目还有:

  • 【345. 反转字符串中的元音字母】

四、141. 环形链表

给定一个链表,判断链表中是否有环。为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。

  在链表这种数据结构中,采用前文所说的前后指针并不一定有效(例如单向链表),这种情况下,双指针的表现形式为:快慢指针

  快慢指针指的是:设置两个前进方向相同但速度不同的指针。

  本题中,设置每次移动一个单位的慢指针和每次移动两个单位的快指针,那么他们必定会在环内相遇:

JavaScript刷LeetCode拿offer-双指针技巧(上)

参考视频:传送门

  相同类型的题目还有:

  • 【26. 删除排序数组中的重复项】

五、125. 验证回文串

给定一个字符串,验证它是否是回文串,只考虑字母和数字字符,可以忽略字母的大小写。说明:本题中,我们将空字符串定义为有效的回文串。

  回文字符串问题是双指针的经典应用,同时也是面试题中的常客。

JavaScript刷LeetCode拿offer-双指针技巧(上)

六、27. 移除元素

给定一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,返回移除后数组的新长度。不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

  显而易见的解决方法是通过 while + splice 处理,但是 splice 操作方法是非常耗时的,每次删除元素之后,需要重排数组中的元素,具有相同副作用的操作方法还有 unshift 和 shift 。 (具体可以查看 V8 源码)

  相比较下,pop 和 push 则是非常快的操作方法,这里可以采用双指针 + pop 操作方法,进一步优化时间复杂度:

JavaScript刷LeetCode拿offer-双指针技巧(上)

写在最后

  算法作为计算机的基础学科,用 JavaScript 刷,一点也不丢人ε=ε=ε=┏(゜ロ゜;)┛。

  本系列文章会分别给出一种算法的3种难度的总结篇(简单难度,中等难度以及困难难度)。在简单难度中,会介绍该算法的基本知识与实现,另外两个难度,着重讲解解题的思路。

  如果本文对您有所帮助,可以点赞或者关注来鼓励博主。

点赞
收藏
评论区
推荐文章
待兔 待兔
1年前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Karen110 Karen110
4年前
一篇文章带你了解JavaScript日期
日期对象允许您使用日期(年、月、日、小时、分钟、秒和毫秒)。一、JavaScript的日期格式一个JavaScript日期可以写为一个字符串:ThuFeb02201909:59:51GMT0800(中国标准时间)或者是一个数字:1486000791164写数字的日期,指定的毫秒数自1970年1月1日00:00:00到现在。1\.显示日期使用
似梦清欢 似梦清欢
3年前
查找算法
顺序查找顺序查找又称为线性查找,对线性表和链表都适用。线性表可以通过数组下标递增来顺序扫描每个元素,链表可以通过next指针依次扫描每一个元素。:::tip指针实现顺序表时,顺序表中是指针时,在定义顺序表的结构体后,需要对顺序表初始化,初始化时为指针申请堆
Caomeinico Caomeinico
4年前
二叉树展开为链表
给你二叉树的根结点root,请你将它展开为一个单链表:展开后的单链表应该同样使用TreeNode,其中right子指针指向链表中下一个结点,而左子指针始终为null。展开后的单链表应该与二叉树先序遍历顺序相同。classSolutionpublicvoidflatten(TreeNoderoot)if
BichonCode BichonCode
5年前
双指针问题
一、双指针之左右指针相关题目1.1题目要求:给定一个升序排列的整数数组,找到两个数,使它们的和等于给定的数,有且仅有一个满足条件的解,返回索引。题目分析:需要两个指针,一个指向开头,一个指向末尾,然后向中间遍历,如果指向的两个数相加正好等于target的话,直接返回两个指针的位置即可,若小于target,左指针右移一位,若大于target,右
Kevin501 Kevin501
5年前
Go语言中new()和make()的区别
1.Go语言中的值类型和引用类型值类型:int,float,bool,string,struct和数组(数组要特别注意,别搞混了)变量直接存储值,分配栈区的内存空间,这些变量所占据的空间在函数被调用完后会自动释放。引用类型:slice,map,chan和值类型对应的指针变量存储的是一个地址(或者理解为指针),指针指向内存中真
Wesley13 Wesley13
4年前
Oracle一张表中实现对一个字段不同值和总值的统计(多个count)
需求:统计WAIT\_ORDER表中的工单总数、未处理工单总数、已完成工单总数、未完成工单总数。表结构:为了举例子方便,WAIT\_ORDER表只有两个字段,分别是ID、STATUS,其中STATUS为工单的状态。1表示未处理,2表示已完成,3表示未完成总数。 SQL:  1.SELECT   2
Wesley13 Wesley13
4年前
C++学习(十八)(C语言部分)之 指针2
指针1、指针的概述指针是什么?指针是一个地址是一个常量int整型intaa是变量指针用来做什么?方便使用数组或者字符串像汇编语言一样处理内存地址2、指针变量什么是指针变量?是一个可以存储地址的一个“容器”经常会吧指针变量读作指针后面吧地址当做“指针”吧存储地址的变量叫做“指针变量”
Wesley13 Wesley13
4年前
C语言之指针
  学过编程语言的童鞋们都知道指针是C语言的精髓,学好了指针就等于学好了C语言,它能够直接对物理地址进行访问,具有双重功能,是嵌入式设计中必不可少的一门语言。C语言功能强大的主要原因就是具有指针结构。指针是一种特殊的数据类型,直接指向目标的存储地址,实现直接访问对象存储空间功能。指针到底是什么    计算机的内存被划分为多个存储单
深度学习 深度学习
7个月前
LeetCode 2576题解:双指针法求解最多标记下标(排序+贪心策略)
一、题目解读2576题要求在一个整数中寻找最多可标记的下标对:若nums法”的组合思路:1.排序预处理:对原数组nums进行升序排序,确保相同元素聚集,便于后续配对。2.双划分:将排序后的数组分为左右两半(左指针left0,右指针rightn/2),从
贾蔷 贾蔷
7个月前
力扣15题三数之和解法(C++双指针算法详解)
一、题目解读15题()要求在一个包含n个整数的中,找出所有三个数之和为0的组合,且每个组合的元素不能重复。题目考察数组遍历、与技巧的结合,是经典的多问题,对时间复杂度优化有较高要求。二、解题思路采用“双指针”策略:首先对原数组排序,然后固定第一个数,通过左