iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >有哪些链表的小技巧
  • 729
分享到

有哪些链表的小技巧

2023-06-16 01:06:34 729人浏览 独家记忆
摘要

本篇内容介绍了“有哪些链表的小技巧”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!链表链表是数据结构里一个很基础但是又很爱考的线性结构,链表的

本篇内容介绍了“有哪些链表的小技巧”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

链表

链表是数据结构里一个很基础但是又很爱考的线性结构,链表的操作相对来说比较简单,但是非常适合考察面试者写代码的能力,以及对 corner case 的处理,还有指针的应用很容易引起 NPE (null pointer exception)。综合以上原因,链表在面试中很重要。

提到链表就不得不提数组,它和数组可以说是数据结构的基础,那么它们最主要的区别在于:

  •  数组在物理内存上必须是连续的

  •  链表在物理内存上不需要连续,通过指针连接

所以数组最好的性质就是可以随机访问 random access,有了 index,可以 O(1) 的时间访问到元素。

而链表因为不连续,所以无法 O(1) 的时间定位任意一个元素的位置,那么就只能从头开始遍历。

这就造成了它们之间增删改查上效率的不同。

除此之外,链表本身的结构与数组也是完全不同的。

LinkedList 是由 Listnode 来实现的:

class ListNode {    int value;    ListNode next;  }

结构上长这样:

有哪些链表的小技巧

这是单向链表,那还有的链表是双向链表,也就是还有一个 previous pointer 指向当前 node 的前一个 node:

class ListNode {    int value;    ListNode next;    ListNode prev;  }

有哪些链表的小技巧

其实链表相关的题目没有很难的,套路也就这么几个,其中最常考最基础的题目是反转链表,听说微软可以用这道题电面刷掉一半的 candidate,两种方法一遍 bug free 还是不容易的。文章之前已经写过了,点击这里直达复习。

今天我们来说链表中最主要的 2 个技巧:双指针法和 dummy node,相信看完本文后,链表相关的绝大多数题目你都能搞定啦。

双指针法

双指针法在很多数据结构和题型中都有应用,在链表中用的最多的还是快慢指针。

顾名思义,两个指针一个走得快,一个走得慢,这样的好处就是以不同的速度遍历链表,方便找到目标位置。

常见的问题比如找一个链表的中点,或者判断一个链表是否有环。

例 1:找中点

有哪些链表的小技巧

这题就是给一个链表,然后找到它的中点,如果是奇数个很好办,如果是偶数个,题目要求返回第二个。

比如:

1 -> 2 -> 3 -> 4 -> 5 -> NULL,需要返回 3 这个 ListNode;

1 -> 2 -> 3 -> 4 -> 5 -> 6 -> NULL,需要返回 4 这个 ListNode。

但其实吐槽一下,如果真的要设计一个这样的 api,我更倾向于选择返回偶数个中的第一个中点。

为什么呢?

算法题都是工业生产中一些问题的抽象。比如说我们找中点的目的是为了把这个链表断开,那么返回了 3,我可以断开 3 和 4;但是返回了 4,单向链表我怎么断开 4 之前的地方呢?还得再来一遍,麻烦。

Solution

方法一、

这题最直观的解法就是可以先求这个链表的长度,然后再走这个长度的一半,得到中点。

class Solution {      public ListNode middleNode(ListNode head) {          if(head == null) {            return null;          }          int len = 0;          ListNode current = head;          while(current != null) {              len++;              currentcurrent = current.next;          }          len /= 2;          ListNode result = head;          while(len > 0) {              resultresult = result.next;              len--;          }          return result;      }  }

方法二、快慢指针

我们用两个指针一起来遍历这个链表,每次快指针走 2 步,慢指针走 1 步,这样当快指针走到头的时候,慢指针应该刚好在链表的中点。

class Solution {      public ListNode middleNode(ListNode head) {          ListNode slow = head;          ListNode fast = head;          while(fast != null && fast.next != null) {              slowslow = slow.next;              fastfast = fast.next.next;          }          return slow;      }  }

这两个方法孰优孰劣呢?

网上很多说什么方法一过了两遍链表,方法二只过了一遍。

但其实,但是方法二用了两个指针来遍历,所以两个方法过的遍数都是一样的。

它们最大的区别是:

方法一是 offline alGorithm,方法二是 online algorithm。

公司里的数据量是源源不断的,比如电商系统里总有客户在下单,社交软件里的好友增长是一直在涨的,这些是数据流 data stream,我们是无法计算数据流的长度的。

那么 online algorithm 能够给时刻给出当前的结果,不用说等数据全部录入完成后,实际上也录不完。。这是 online algorithm 比 offline algorithm 大大的优势所在。

更多的解释大家可以参考 stack overflow 的这个问题,链接在文末。

有哪些链表的小技巧

例 2:判断单链表是否有环

有哪些链表的小技巧

思路:快慢指针一起从 head 出发,每次快指针走 2 步,慢指针只走 1 步,如果存在环,那么两个指针一定会相遇。

这题是典型的龟兔赛跑,或者说在操场跑圈时,跑的快的同学总会套圈跑的慢的。

public class Solution {      public boolean hasCycle(ListNode head) {          ListNode slow = head;          ListNode fast = head;          while(fast != null && fast.next != null) {              slowslow = slow.next;              fastfast = fast.next.next;              if(slow == fast) {                  return true;              }          }          return false;      }  }

这题有个升级版,就是要求返回环的起点。

例 3:返回有环链表的环的起点

有哪些链表的小技巧

这题我感觉不全是个算法题了,还是个数学题哈哈。

先摆出结论:

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2.  快慢指针从链表头开始走,相遇的那点,记为 M;

  3.  再用 2 个指针,一个从头开始走,一个从 M 开始走,相遇点即为 cycle 的起点。

我们先看抽象出来的图:

假设快慢指针在 M 点第一次相遇,

这里我们设 3 个变量来表示这个链表里的几个重要长度:

  •  X:从链表头到环的起点的长度;

  •  Y:从环的起点到 M 点的长度;

  •  Z:从 M 点到环的起点的长度。

注意:因为环是有方向的,所以 Y 并不是 Z。

那其实我们唯一知道的关系就是:快慢指针在 M 点第一次相遇。这也是我们最初假设的关系。

而快慢指针有一个永远不变的真理:快指针走的长度永远是慢指针走的长度的 2 倍。

相遇时快慢指针分别走了多少的长度呢?

  •  快指针:X+ Y + 假设走了 k 圈

  •  慢指针:X + Y

那么我们就可以用这个 2 倍的关系,列出下列等式:

2 * (X + Y) = X + Y + kL

所以 X + Y = kL

而我们注意到:Y + Z = L,那么就能得出 X = Z。

所以当两个指针,一个从头开始走,一个从 M 点开始走时,相遇那点就是环的起点,证毕。

来看下代码吧:

public class Solution {    public ListNode detectCycle(ListNode head) {      ListNode slow = head;      ListNode fast = head;      while (fast != null && fast.next != null) {        slowslow = slow.next;        fastfast = fast.next.next;        if (slow == fast) {          ListNode x = head;         ListNode y = slow;          while(x != y) {            xx = x.next;            yy = y.next;          }          return x;        }      }      return null;    }  }

这题还有个应用,就是找一个特定数组里重复的数字,这里就不展开了,大家感兴趣的去做一下吧~

有哪些链表的小技巧

接下来我们聊聊 dummy node 这个技巧。

Dummy node

Dummy 的中文是“假”的意思,dummy node 大概可以翻译成虚拟节点?有更地道的说法的话还请大家在评论区告诉我呀~

一般来说,dummy node 的用法是在链表的真实 head 的前面加一个指向这个 head 的节点,目的是为了方便操作 head。

对于链表来说,head 是一个链表的灵魂,因为无论是查询还是其他的操作都需要从头开始,俗话说擒贼先擒王嘛,抓住了一个链表的头,就抓住了整个链表。

所以当需要对现有链表的头进行改动时,或者不确定头部节点是哪个,我们可以预先加一个 dummyHead,这样就可以灵活处理链表中的剩余部分,最后返回时去掉这个“假头”就好了。

很多时候 dummy node 不是必须,但是用了会很方便,减少 corner case 的讨论,所以还是非常推荐使用的。

光说不练假把式,我们直接来看题~

例 4:合并两个排好序的链表

有哪些链表的小技巧

这题有很多种解法,比如最直观的就是用两个指针,然后比较大小,把小的接到最终的结果上去。

但是有点麻烦的是,最后的结果不知道到底谁是头啊,是哪个链表的头作为了最终结果的头呢?

这种情况就非常适合用 dummy node。

先用一个虚拟的头在这撑着,把整个链表构造好之后,再把这个假的剔除。

来看代码~

class Solution {    public ListNode mergeTwoLists(ListNode l1, ListNode l2) {      if (l1 == null) {        return l2;      }      if (l2 == null) {        return l1;      }      ListNode dummy = new ListNode(0);      ListNode ptr = dummy;      while (l1 != null && l2 != null) {        if (l1.val < l2.val) {          ptr.next = l1;          l1l1 = l1.next;        } else {          ptr.next = l2;          l2l2 = l2.next;        }        ptrptr = ptr.next;      }      if (l1 == null) {        ptr.next = l2;     } else {        ptr.next = l1;      }      return dummy.next;    }  }

这题也有升级版,就是合并 k 个排好序的链表。本质上也是一样的,只不过需要重写一下比较器就好了。

例 5:删除节点

有哪些链表的小技巧

这道题的意思是删除链表中某个特定值的节点,可能有一个可能有多个,可能在头可能在尾。

如果要删除的节点在头的时候,新链表的头就不确定了,也有可能是个空的。。此时就很适合用 dummy node 来做,规避掉这些 corner case 的讨论。

那这题的思路就是:用 2 个指针

  •  prev:指向当前新链表的尾巴

  •  curr:指向当前正在遍历的 ListNode

如果 curr == 目标值,那就直接移到下一个;

如果 curr != 目标值,那就把 prev 指向它,接上。

这题需要注意的是,最后一定要把 prev.next 指向 null,否则如果原链表的尾巴是目标值的话,还是去不掉。

代码如下:

class Solution {    public ListNode removeElements(ListNode head, int val) {      ListNode dummy = new ListNode(0);      ListNode prev = dummy;      ListNode curr = head;      while(curr != null) {        if (curr.val != val) {          prev.next = curr;          prevprev = prev.next;        }        currcurr = curr.next;      }      prev.next = null;      return dummy.next;    }  }

“有哪些链表的小技巧”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 有哪些链表的小技巧

本文链接: https://www.lsjlt.com/news/282086.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 有哪些链表的小技巧
    本篇内容介绍了“有哪些链表的小技巧”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!链表链表是数据结构里一个很基础但是又很爱考的线性结构,链表的...
    99+
    2023-06-16
  • SEO外链发布的实用小技巧有哪些
    这篇文章主要介绍了SEO外链发布的实用小技巧有哪些,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1、软文发布:之前有个博友问过我,软文发布可以吗,A5和CHINAZ只允许发布...
    99+
    2023-06-10
  • PyCharm的小技巧有哪些
    PyCharm的小技巧有哪些,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。PyCharm小技巧,帮助大家提升工作效率!# 0. PyCharm 常用快捷键# 1. 查看使用...
    99+
    2023-06-02
  • Vue的小技巧有哪些
    这篇文章主要介绍“Vue的小技巧有哪些”,在日常操作中,相信很多人在Vue的小技巧有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Vue的小技巧有哪些”的疑惑有所帮助!接...
    99+
    2024-04-02
  • mysql小技巧有哪些
    这篇文章主要介绍mysql小技巧有哪些,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完! 1.  查看历史操作记录 1.1   linux操作系统查看历...
    99+
    2024-04-02
  • ES6小技巧有哪些
    小编给大家分享一下ES6小技巧有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1. 强制要求参数ES6提供了默认参数值机制,...
    99+
    2024-04-02
  • JavaScript有哪些小技巧
    这篇文章主要为大家展示了“JavaScript有哪些小技巧”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“JavaScript有哪些...
    99+
    2024-04-02
  • 有哪些Python小技巧
    这篇文章主要讲解了“有哪些Python小技巧”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“有哪些Python小技巧”吧!集合开发人员常常忘记 Python 也有集合数据类型,大家都喜欢使用列...
    99+
    2023-06-15
  • Google Chrome的小技巧有哪些
    Google Chrome的小技巧有哪些,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Google Chrome(开源项目Chromium) 是一个由Googl...
    99+
    2023-06-17
  • JavaScript的小技巧有哪些呢
    这期内容当中小编将会给大家带来有关JavaScript的小技巧有哪些呢,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。1、过滤唯一值Set类型是在ES6中新增的,它类似于数...
    99+
    2024-04-02
  • 写JavaScript的小技巧有哪些
    本篇内容介绍了“写JavaScript的小技巧有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. a...
    99+
    2024-04-02
  • 写Python有哪些小技巧
    本篇内容介绍了“写Python有哪些小技巧”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 反转字符串虽然看似是很基础的操作,但是用cha...
    99+
    2023-06-16
  • CSS的实用小技巧有哪些
    这篇文章给大家分享的是有关CSS的实用小技巧有哪些的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。1. 打字效果代码实现:<div class="wrap...
    99+
    2024-04-02
  • 必备的CSS小技巧有哪些
    这篇文章主要介绍了必备的CSS小技巧有哪些的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇必备的CSS小技巧有哪些文章都会有所收获,下面我们一起来看看吧。 不久之前Firefo...
    99+
    2024-04-02
  • PHP有哪些常用的小技巧
    这篇文章主要为大家展示了“PHP有哪些常用的小技巧”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“PHP有哪些常用的小技巧”这篇文章吧。1、命名<input type='c...
    99+
    2023-06-17
  • 常用的C++小技巧有哪些
    小编给大家分享一下常用的C++小技巧有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1、头文件是引用<iostream.h>还是<iostr...
    99+
    2023-06-22
  • 使用css的小技巧有哪些
    这篇文章主要为大家展示了“使用css的小技巧有哪些”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“使用css的小技巧有哪些”这篇文章吧。 小编做前端项目也有一段...
    99+
    2024-04-02
  • 常用的CSS小技巧有哪些
    本篇内容介绍了“常用的CSS小技巧有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1.清除图片下方出现...
    99+
    2024-04-02
  • 有哪些特别的CSS小技巧
    这篇文章主要讲解了“有哪些特别的CSS小技巧”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“有哪些特别的CSS小技巧”吧!1.在CSS中用attr()显示HT...
    99+
    2024-04-02
  • 有哪些使用Java的小技巧
    本篇内容介绍了“有哪些使用Java的小技巧”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!使用stream实现list转map普通:对于lis...
    99+
    2023-06-15
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作