动图:删除链表的倒数第 N 个结点

开发 前端
本文主要介绍一道面试中常考链表删除相关的题目,即 leetcode 19. 删除链表的倒数第 N 个结点。采用 双指针 + 动图 的方式进行剖析,供大家参考,希望对大家有所帮助。

 [[393057]]

本文转载自微信公众号「程序员小熊」,作者Dine。转载本文请联系程序员小熊公众号。   

本文主要介绍一道面试中常考链表删除相关的题目,即 leetcode 19. 删除链表的倒数第 N 个结点。采用 双指针 + 动图 的方式进行剖析,供大家参考,希望对大家有所帮助。

删除链表的倒数第 N 个结点

给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。

进阶:你能尝试使用一趟扫描实现吗?

解题思路

在链表中要删除某个节点 nodeB,必须先找到 nodeB 的前一节点 nodeA ,再将 nodeA 指向 nodeB 的下一节点 nodeC ,从而实现节点 nodeB 的删除。

例如要删除链表 L(1->2->3->4->5) 中 值为 3 的节点,首先得找到该节点的前一节点(值为 2 的节点),才能实现该节点的删除,如下图示:

题目要求删除 倒数第 n 个 节点,所以首先得找到 该节点的前一节点 ,但由于不知道 整个链表的长度,因此不知道 待删除的节点是正数的第几个节点,所以很难从头节点开始遍历时删除掉这个节点。

思路一

先遍历一遍链表,获取整个链表的长度;假设整个链表的长度为 l,则可知要删除的节点为第 l - n + 1 个节点;再遍历一遍,删除倒数第 n 个节点。例如链表 L 为 1->2->3->4->5,n = 3,则要删除的节点为 第 5 - 3 + 1 = 3 个节点 。

思路二

尽管思路一可行,但是需要 遍历链表两遍,不够简洁,而且题目的 进阶 中也提到尝试使用一趟扫描实现,因此本文采用 双指针 的策略,实现通过一次扫描删除 倒数第 n 个节点 。

在上一期链表相关专题 虚拟头节点秒杀链表问题 中提到 增加虚拟头节点 的策略解决链表问题,增加虚拟头节点的 好处 在于:不需要单独考虑头节点,这样对头节点的处理就像跟其它节点一样。本文也同样采用这种策略。

举栗

以链表 1->2->3->4->5,n = 3 为栗,如下如示。

按照上面分析,先要找到 倒数第 3 个节点的前一节点,即值为 2 的节点;

增加虚拟头节点

值为 2 的节点是 倒数第 4 个节点(后往前数),增加两指针 fast/slow,分别指向最后一个元素(NULL)和上图中 target 的位置;

此时 fast 跟 slow 之间的间距是固定(n = 3)的,找到 target(slow)后,只需要删除其下一节点即可,但 slow 指向的节点前面有多少个节点该如何确定呢?

由于当前已知 fast 和 slow 指向节点之间的长度是固定的,只需要将这两个指针向前挪,直到 slow 挪到虚拟头节点(值为 0)的位置,此时 fast 指向值为 4 的节点的位置,fast 只需要由虚拟头节点的位置 右移 n + 1 = 4 即可。如下图示:

当 fast 右移至值为 4 的节点时,指针 slow 和 fast 同时右移,直至 fast 移到 NULL,此时 slow 刚好到 target 位置,即指向 倒数第 n + 1 个节点,如下图示。

Show me the Code

c++

  1. ListNode* removeNthFromEnd(ListNode* head, int n) { 
  2.     ListNode* dummyHead = new ListNode(0); 
  3.     dummyHead->next = head; 
  4.     ListNode *slow = dummyHead, *fast = dummyHead; 
  5.     for (int i = 0; i < n + 1; ++i) { 
  6.         fast = fast->next
  7.     } 
  8.  
  9.     while (fast != NULL) { 
  10.         slow = slow->next
  11.         fast = fast->next
  12.     } 
  13.  
  14.     ListNode* delNode = slow->next
  15.     slow->next = delNode->next
  16.     delete delNode; 
  17.  
  18.     ListNode* retNode = dummyHead->next
  19.     delete dummyHead; 
  20.  
  21.     return retNode; 

java

  1. ListNode removeNthFromEnd(ListNode head, int n) { 
  2.     ListNode dummyHead = new ListNode(0); 
  3.     dummyHead.next = head; 
  4.     ListNode slow = dummyHead, fast = dummyHead; 
  5.     for (int i = 0; i < n + 1; ++i) { 
  6.         fast = fast.next
  7.     } 
  8.  
  9.     while (fast != null) { 
  10.         slow = slow.next
  11.         fast = fast.next
  12.     } 
  13.  
  14.     slow.next = slow.next.next
  15.     return dummyHead.next

 

责任编辑:武晓燕 来源: 程序员小熊
相关推荐

2021-08-10 07:57:03

算法链表倒数

2022-01-17 09:23:02

LeetCode删除链表算法

2021-02-03 13:23:42

链表倒数结点

2020-10-19 13:27:19

链表倒数结点

2022-06-01 06:58:41

节点链表倒数

2023-04-17 07:33:11

反转链表移除链表

2012-06-19 14:23:04

云计算中国

2021-02-04 08:18:53

LeetCode链表

2012-02-17 09:45:04

网速手机

2012-02-17 09:43:13

手机网速移动互联

2010-11-15 10:49:23

求职

2012-08-10 10:53:03

云计算BSA商业软件联盟

2012-06-18 10:07:17

云计算实力榜

2021-08-26 10:07:25

数组前端元素

2022-03-07 11:03:08

大数据检测谷歌

2018-03-01 13:32:28

宏碁游戏本PC行业

2021-09-22 22:57:41

手机流量通信

2011-08-08 10:53:55

宝德PR2510N云计算

2021-04-27 08:31:06

event loopJavaScriptsetTimeout函

2012-02-21 17:17:51

手机网速网速
点赞
收藏

51CTO技术栈公众号