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

一、前言

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

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

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

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

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

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

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

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

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

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

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

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

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

在这里插入图片描述

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

三、344. 反转字符串

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

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

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

在这里插入图片描述

  相同类型的题目还有:

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

四、141. 环形链表

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

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

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

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

在这里插入图片描述

参考视频:传送门

  相同类型的题目还有:

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

五、125. 验证回文串

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

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

在这里插入图片描述

六、27. 移除元素

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

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

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

在这里插入图片描述

写在最后

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

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

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

写代码使我快乐

20 声望
1 粉丝
0 条评论
推荐阅读
JavaScript刷LeetCode拿offer-双指针
一、前言  一般情况下,遍历数组(或者字符串)操作,都是采用单指针从前往后或者从后往前依次访问数组(或者字符串)中的元素。  而对于以下情况,只采用单指针处理,则会徒增时间复杂度和空间复杂度:例如:找...

hellocoder2028阅读 182

安全地在前后端之间传输数据 - 「3」真的安全吗?
在「2」注册和登录示例中,我们通过非对称加密算法实现了浏览器和 Web 服务器之间的安全传输。看起来一切都很美好,但是危险就在哪里,有些人发现了,有些人嗅到了,更多人却浑然不知。就像是给门上了把好锁,还...

边城31阅读 7.2k评论 5

封面图
涨姿势了,有意思的气泡 Loading 效果
今日,群友提问,如何实现这么一个 Loading 效果:这个确实有点意思,但是这是 CSS 能够完成的?没错,这个效果中的核心气泡效果,其实借助 CSS 中的滤镜,能够比较轻松的实现,就是所需的元素可能多点。参考我们...

chokcoco20阅读 2.1k评论 2

在前端使用 JS 进行分类汇总
最近遇到一些同学在问 JS 中进行数据统计的问题。虽然数据统计一般会在数据库中进行,但是后端遇到需要使用程序来进行统计的情况也非常多。.NET 就为了对内存数据和数据库数据进行统一地数据处理,发明了 LINQ (L...

边城17阅读 1.9k

封面图
【已结束】SegmentFault 思否写作挑战赛!
SegmentFault 思否写作挑战赛 是思否社区新上线的系列社区活动在 2 月 8 日 正式面向社区所有用户开启;挑战赛中包含多个可供作者选择的热门技术方向,根据挑战难度分为多个等级,快来参与挑战,向更好的自己前进!

SegmentFault思否20阅读 5.6k评论 10

封面图
过滤/筛选树节点
又是树,是我跟树杠上了吗?—— 不,是树的问题太多了!🔗 相关文章推荐:使用递归遍历并转换树形数据(以 TypeScript 为例)从列表生成树 (JavaScript/TypeScript) 过滤和筛选是一个意思,都是 filter。对于列表来...

边城18阅读 7.7k评论 3

封面图
Vue2 导出excel
2020-07-15更新 excel导出安装 {代码...} src文件夹下新建一个libs文件夹,新建一个excel.js {代码...} vue页面中使用 {代码...} ===========================以下为早期的文章今天在开发的过程中需要做一个Vue的...

原谅我一生不羁放歌搞文艺14阅读 19.9k评论 9

写代码使我快乐

20 声望
1 粉丝
宣传栏