从快慢指针到倒数查找:优雅解决链表倒数问题

从生活场景说起

想象你在一个漫长的队伍中,想知道自己距离队尾还有多少人。一个巧妙的方法是:让你的朋友从你所在位置往后数N步,然后你和朋友一起向后走。当朋友走到队尾时,你的位置就正好是倒数第N个。这个生活中的小技巧,正是我们今天要探讨的链表算法的灵感来源。

问题描述

LeetCode第19题"删除链表的倒数第N个结点"要求:给你一个链表的头节点 head 和一个整数 n ,请你删除链表的倒数第 n 个结点,并且返回链表的头结点。

例如:

输入:1 → 2 → 3 → 4 → 5, n = 2
输出:1 → 2 → 3 → 5
解释:删除倒数第2个节点(值为4) 输入:1 → 2, n = 2
输出:2
解释:删除倒数第2个节点(值为1) 输入:1, n = 1
输出:空链表
解释:删除唯一的节点

初步思路:两次遍历法

最直观的解法是先遍历一遍链表得到长度,然后再遍历一次删除目标节点:

public ListNode removeNthFromEnd(ListNode head, int n) {
// 第一次遍历,计算链表长度
int length = 0;
ListNode current = head;
while (current != null) {
length++;
current = current.next;
} // 如果要删除的是头节点
if (length == n) {
return head.next;
} // 找到待删除节点的前一个节点
current = head;
for (int i = 0; i < length - n - 1; i++) {
current = current.next;
} // 执行删除操作
current.next = current.next.next; return head;
}

优化解法:快慢指针一次遍历

就像我们在队伍中的例子,我们可以用快慢指针在一次遍历内解决问题:

public ListNode removeNthFromEnd(ListNode head, int n) {
// 创建哨兵节点,统一处理头节点的删除
ListNode dummy = new ListNode(0);
dummy.next = head; // 初始化快慢指针
ListNode fast = dummy;
ListNode slow = dummy; // 快指针先走n+1步(多走一步是为了让慢指针停在待删除节点的前一个位置)
for (int i = 0; i <= n; i++) {
fast = fast.next;
} // 快慢指针同步移动
while (fast != null) {
fast = fast.next;
slow = slow.next;
} // 执行删除操作
slow.next = slow.next.next; return dummy.next;
}

图解过程

例子:删除倒数第2个节点
1) 初始状态:
dummy → 1 → 2 → 3 → 4 → 5
S,F 2) 快指针先走n+1步:
dummy → 1 → 2 → 3 → 4 → 5
S F 3) 同步移动直到快指针到末尾:
dummy → 1 → 2 → 3 → 4 → 5
S F 4) 删除slow.next节点:
dummy → 1 → 2 → 3 → 5

深入理解快慢指针解法

为什么这个方法能工作?让我们仔细分析:

  1. 快指针先走n+1步,与慢指针产生n+1的距离
  2. 当快指针到达末尾(null)时,慢指针正好在待删除节点的前一个位置
  3. 这保证了我们总能找到待删除节点的前驱节点,便于执行删除操作

复杂度分析

两次遍历法:

  • 时间复杂度:O(L),需要两次遍历
  • 空间复杂度:O(1)
  • 优点:直观易懂
  • 缺点:需要两次遍历

快慢指针法:

  • 时间复杂度:O(L),只需一次遍历
  • 空间复杂度:O(1)
  • 优点:一次遍历即可完成,更优雅
  • 缺点:需要理解快慢指针的原理

技巧总结

  1. 哨兵节点的使用

    • 统一了头节点的处理
    • 避免了额外的边界检查
  2. 快慢指针的设计

    • 快指针先走n+1步的巧妙设计
    • 同步移动直至快指针到达末尾
  3. 边界情况的处理

    • 链表长度等于n
    • 只有一个节点
    • n等于链表长度

实际应用延伸

这种快慢指针的思想在实际开发中有很多应用:

  1. 缓存淘汰算法
  2. 流式数据的滑动窗口处理
  3. 实时数据处理中的延迟计算

小结与思考

通过这个问题,我们学到了:

  1. 如何用空间换时间(两次遍历)
  2. 如何用巧妙的算法优化空间(快慢指针)
  3. 哨兵节点的实用价值
  4. 如何优雅处理链表的边界情况

当我们遇到类似的"倒数"问题时,可以考虑:

  1. 是否可以用快慢指针解决?
  2. 是否需要哨兵节点简化处理?
  3. 如何在一次遍历中完成任务?

记住:有时看似复杂的问题,用合适的思维方式就能找到优雅的解决方案。


作者:忍者算法

公众号:忍者算法

我准备了一份刷题清单,以及这些题目的详细题解,覆盖了绝大部分常见面试题。我可以很负责任地说,只要你把这些题真正掌握了,80%的算法面试都能遇到相似题目。公众号回复【刷题清单】获取~

【忍者算法】从快慢指针到倒数查找:优雅解决链表倒数问题|LeetCode第19题"删除链表的倒数第N个结点"的更多相关文章

  1. Leetcode(19)-删除链表的倒数第N个节点

    给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点. 示例: 给定一个链表: 1->2->3->4->5, 和 n = 2. 当删除了倒数第二个节点后,链表变为 ...

  2. LeetCode第[19]题(Java):Remove Nth Node From End of List(删除链表的倒数第N个节点)

    题目:删除链表的倒数第N个节点 难度:Medium 题目内容: Given a linked list, remove the n-th node from the end of list and r ...

  3. 【LeetCode题解】19_删除链表的倒数第N个节点(Remove-Nth-Node-From-End-of-List)

    目录 描述 解法:双指针 思路 Java 实现 Python 实现 复杂度分析 更多 LeetCode 题解笔记可以访问我的 github. 描述 给定一个链表,删除链表的倒数第 n 个节点,并且返回 ...

  4. LeetCode 19:删除链表的倒数第N个节点 Remove Nth Node From End of List

    给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点. Given a linked list, remove the n-th node from the end of list and ...

  5. 【LeetCode 19】删除链表的倒数第N个节点

    题目链接 [题解] 经典的一道题. 让p1指向链表的第一个元素. 让p2指向链表的第二个元素. 然后让他们俩同时往后移动. 直到p2到达链表的尾巴. 这时p1和p2之间总是隔了n-1个元素. 所以p1 ...

  6. 19。删除链表倒数第N个节点

    class ListNode: def __init__(self, val=0, next=None): self.val = val self.next = next# 这道题还是很简单的,我们只 ...

  7. 【LeetCode】19. Remove Nth Node From End of List 删除链表的倒数第 N 个结点

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 个人公众号:负雪明烛 本文关键词:链表, 删除节点,双指针,题解,leetcode, 力扣 ...

  8. Leetcode算法系列(链表)之删除链表倒数第N个节点

    Leetcode算法系列(链表)之删除链表倒数第N个节点 难度:中等给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点.示例:给定一个链表: 1->2->3->4-&g ...

  9. 打败算法 —— 删除链表的倒数第n个结点

    本文参考 出自LeetCode上的题库 -- 删除链表的倒数第n个结点,官方的双指针解法没有完全符合"只遍历一遍链表"的要求,本文给出另一种双指针解法 https://leetco ...

  10. 【算法训练营day4】LeetCode24. 两两交换链表中的结点 LeetCode19. 删除链表的倒数第N个结点 LeetCode面试题 02.07. 链表相交 LeetCode142. 环形链表II

    [算法训练营day4]LeetCode24. 两两交换链表中的结点 LeetCode19. 删除链表的倒数第N个结点 LeetCode面试题 02.07. 链表相交 LeetCode142. 环形链表 ...

随机推荐

  1. VS Code 变身小霸王游戏机!

    在韩老师的<Visual Studio Code 权威指南>一书中,我向大家推荐了许多好用的插件,其中也不乏许多摸鱼插件,刷知乎.炒股票.看电影.听音乐.追番.看小说,一应俱全. 今天,就 ...

  2. 关于在Rocky linux下安装dotnet sdk不成功的问题

    Rocky Linux 9,运行 dnf install -y dotnet-sdk-6.0 一切正常,运行起来非常顺利,安装完毕.但是非常诡异,运行 dotnet --list-sdks dotne ...

  3. 为什么 Llama 3.3 70B 比 GPT-4o 和 Claude 3.5 Sonnet 更优秀

    过去七天的 AI 新闻如狂风暴雨般涌来,AI 世界发生了许多重大变化.在这篇文章中,我们将深入探讨来自 Llama 3.3 70B.GPT-4o 和 Claude 3.5 Sonnet 等主要参与者的 ...

  4. Qt音视频开发20-vlc内核动态保存录像文件(不需要重新编译源码)

    一.前言 在vlc默认提供的保存文件方式中,通过打开的时候传入指定的参数来保存文件,直到关闭播放生成文件,这种方式简单暴力,但是不适用大部分的场景,大部分时候需要的是提供开始录制和停止录制的功能,也就 ...

  5. [转]GMM:高斯混合模型讲义(台湾)

  6. Jetbrain Fleet体验版linxu环境开发Python初体验

    Jetbrain Fleet体验版linxu环境开发Python初体验 (base) linxu@linxu-PC:~/Projects/FleetProjects$ conda activate d ...

  7. 百度高效研发实战训练营-Step3

    .# 百度高效研发实战训练营-Step3 3.1 质量意识 关于本部分,将从以下几点进行介绍: 3.1.1 质量的基本概念 意识的树立,源于认识的提高. 首先介绍质量的基本概念,主要包括以下内容: 1 ...

  8. Python依赖库的导入、导出 | 解决内网安装模块问题 | Python

    通过在有网的机器A下下载所有的依赖包至package文件夹下: pip3 download -r requirements.txt -d ./package 将依赖包移动至没有网的机器B下,指定依赖包 ...

  9. 「V 曲闲谈」《万物与我同归于寂》——终与始

      死亡, 胃酸逆流 失去最初模样 开端与结局颠倒 脉搏也摇晃   于强烈的动机下选定这首歌,但构思时又发觉,一个十六岁的青年,似乎在这个艰涩的哲学问题上难以开口.   那么先来说说我的动机.为什么我 ...

  10. Solution -「NOI Simu.」记忆

    \(\mathscr{Description}\)   长度为 \(n\) 的序列 \(A\) 初始全 \(0\), \(C=0\). 第 \(i\ge0\) 个时刻对 \(A\) 进行如下变换: \ ...