news 2026/4/23 14:30:49

【每日算法】LeetCode 19. 删除链表的倒数第 N 个结点

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【每日算法】LeetCode 19. 删除链表的倒数第 N 个结点

对前端开发者而言,学习算法绝非为了“炫技”。它是你从“页面构建者”迈向“复杂系统设计者”的关键阶梯。它将你的编码能力从“实现功能”提升到“设计优雅、高效解决方案”的层面。从现在开始,每天投入一小段时间,结合前端场景去理解和练习,你将会感受到自身技术视野和问题解决能力的质的飞跃。------ 算法:资深前端开发者的进阶引擎

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

1. 题目描述

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

示例 1:

输入:head = [1,2,3,4,5], n = 2 输出:[1,2,3,5]

示例 2:

输入:head = [1], n = 2 输出:[1]

示例 3:

输入:head = [1,2], n = 1 输出:[1]

提示:

  • 链表中结点的数目为sz
  • 1 <= sz <= 30
  • 0 <= Node.val <= 100
  • 1 <= n <= sz

2. 问题分析

本题的核心挑战在于“倒数第N个”这个定位要求。单链表无法逆向回溯,因此必须通过正向遍历来找到这个特定位置。一个关键的边界情况是删除的可能是头节点,这需要我们仔细处理。

3. 解题思路

本题主要有两种主流思路,双指针(快慢指针)法是最优解。

3.1 思路一:两次遍历法(计算链表长度)

  1. 第一次遍历:获取链表的长度L
  2. 计算目标位置:需要删除的正数位置为L - n + 1
  3. 第二次遍历:移动到目标位置的前一个节点(即L - n的位置),执行删除操作。
  4. 时间复杂度:O(L),需要遍历链表两次。
  5. 空间复杂度:O(1)。

3.2 思路二:双指针(快慢指针)法【最优解】

  1. 设置哑节点(Dummy Node):在头节点前创建一个虚拟节点,其next指向head。这是处理链表删除问题的常用技巧,可以优雅地统一处理删除头节点和非头节点的情况,避免复杂的条件判断。
  2. 初始化快慢指针fastslow都指向哑节点。
  3. 快指针先行:让fast指针先向前移动n步。此时,fastslow之间相隔n个节点。
  4. 同步移动:同时移动fastslow,直到fast到达链表的末尾(fast.nextnull)。此时,slow恰好指向待删除节点的前一个节点。因为fastslow始终保持n的间距。
  5. 执行删除slow.next = slow.next.next
  6. 返回新头节点:返回dummy.next
  7. 时间复杂度:O(L),只遍历链表一次。
  8. 空间复杂度:O(1)。

为什么双指针法更优?
它不仅时间复杂度相同,而且在一次遍历中完成,逻辑清晰,代码简洁,是面试官最期望看到的解法。

4. 各思路代码实现(JavaScript)

4.1 两次遍历法实现

/** * Definition for singly-linked list. * function ListNode(val, next) { * this.val = (val===undefined ? 0 : val) * this.next = (next===undefined ? null : next) * } *//** * @param {ListNode} head * @param {number} n * @return {ListNode} */varremoveNthFromEnd=function(head,n){// 1. 第一次遍历:计算链表长度letlength=0;letcurrent=head;while(current!==null){length++;current=current.next;}// 2. 计算要删除节点的正数位置consttargetIndex=length-n;// 3. 处理删除头节点的特殊情况if(targetIndex===0){returnhead.next;}// 4. 第二次遍历:找到目标节点的前一个节点current=head;for(leti=0;i<targetIndex-1;i++){current=current.next;}// 5. 执行删除操作current.next=current.next.next;returnhead;};

4.2 双指针法实现【推荐】

/** * Definition for singly-linked list. * function ListNode(val, next) { * this.val = (val===undefined ? 0 : val) * this.next = (next===undefined ? null : next) * } *//** * @param {ListNode} head * @param {number} n * @return {ListNode} */varremoveNthFromEnd=function(head,n){// 1. 创建哑节点,其next指向原头节点constdummy=newListNode(0,head);// 2. 初始化快慢指针letfast=dummy;letslow=dummy;// 3. 快指针先走 n 步for(leti=0;i<n;i++){fast=fast.next;}// 4. 快慢指针同步前进,直到快指针到达链表末尾while(fast.next!==null){fast=fast.next;slow=slow.next;}// 循环结束时,slow指向待删除节点的前一个节点// 5. 删除节点slow.next=slow.next.next;// 6. 返回新头节点(哑节点的next)returndummy.next;};

5. 复杂度与优缺点对比

实现方案时间复杂度空间复杂度优点缺点
两次遍历法O(L), L为链表长度O(1)思路直观,易于理解和实现需要遍历链表两次,效率非最优;处理头节点删除需额外判断
双指针法(哑节点)O(L), L为链表长度O(1)只遍历一次,效率高代码统一简洁,哑节点技巧避免了头节点的特殊判断;面试和工程中的首选方案引入了额外的哑节点,但对空间复杂度无影响,逻辑上稍需理解

6. 总结与实际应用场景

6.1 总结

  1. 哑节点(Dummy Node)技巧:这是解决链表问题的“瑞士军刀”。它在链表头部之前创建一个虚拟节点,使得对头节点的操作(增、删)和对中间节点的操作逻辑完全一致,极大地简化了代码和逻辑判断。
  2. 快慢指针模式:这是链表算法的核心模式之一,不仅用于本题,还广泛应用于检测环形链表(LeetCode 141)寻找链表中点(LeetCode 876)寻找环形链表入口(LeetCode 142)等问题。
  3. 对“倒数第N个”的转化思维:通过快指针先走N步,将“倒数”问题转化为两个指针之间的“固定间隔”问题,这是一种非常巧妙的思维方式。

6.2 前端实际应用场景

  1. 虚拟DOM与Fiber架构:React的Fiber节点通过链表链接。在协调(Reconciliation)过程中,中断和恢复渲染的能力,本质上依赖于对这片“链表森林”的可控遍历。理解指针移动和节点操作,能帮你更深层理解React的调度机制。
  2. 长列表/无限滚动性能优化:在渲染成千上万条数据的列表时(如聊天记录、新闻流),通常采用“窗口化”技术,只渲染可视区域的部分节点。维护这些节点的缓存池、计算哪些节点应该进入或离开可视区,其数据结构的底层逻辑与链表操作息息相关。
  3. 撤销/重做(Undo/Redo)功能:编辑器的历史记录栈可以用双向链表来实现,每个节点保存一个状态。删除历史记录中的某一项,就类似于链表的删除操作。
  4. 路由历史记录管理:浏览器或前端路由库(如React Router)的历史记录管理,其前进、后退、替换等操作,底层都可以用链表模型来理解和设计。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 13:50:07

Ubuntu命令行部署GPT-SoVITS语音合成指南

Ubuntu命令行部署GPT-SoVITS语音合成指南 在远程服务器上做语音合成&#xff0c;最头疼的莫过于没有图形界面——WebUI再炫酷也用不了。这时候&#xff0c;纯命令行就成了唯一的出路。本文记录的就是这样一个实战流程&#xff1a;如何在无GUI的Ubuntu环境中&#xff0c;从零开…

作者头像 李华
网站建设 2026/4/23 12:47:42

LobeChat能否设置额度预警?避免超额支出

LobeChat 能否设置额度预警&#xff1f;避免超额支出 在企业与个人纷纷拥抱大语言模型&#xff08;LLM&#xff09;的今天&#xff0c;AI聊天界面已成为人机交互的核心入口。然而&#xff0c;随着调用量的增长&#xff0c;OpenAI 等云服务按 Token 计费的模式也带来了不可忽视的…

作者头像 李华
网站建设 2026/4/23 11:27:08

LobeChat能否用来练习外语口语?语音输入体验

LobeChat能否用来练习外语口语&#xff1f;语音输入体验 在如今这个语言学习资源触手可及的时代&#xff0c;很多人依然卡在一个最基础的问题上&#xff1a;敢不敢开口说。教材背得滚瓜烂熟&#xff0c;语法掌握得头头是道&#xff0c;可一到真实对话场景就大脑空白、结结巴巴。…

作者头像 李华
网站建设 2026/4/23 12:58:03

Playwright新人笔记学习记录(鉴权2)--Day5

今天自己手动部署了jenkins和Docker&#xff0c;实现CICD&#xff0c;结果发现一个问题&#xff0c;如果单个运行测试类是没有问题的&#xff0c;但是批量运行就会失败。报错信息如下所示&#xff0c;提示定位元素超时了。最后我发现是批量运行时&#xff0c;保存的cookie失效了…

作者头像 李华
网站建设 2026/4/23 12:30:56

Wan2.2-T2V-A14B本地部署全攻略

Wan2.2-T2V-A14B本地部署全攻略 你有没有试过&#xff0c;在一个深夜加班的晚上&#xff0c;突然冒出这样一个念头&#xff1a;如果能用一句话就生成一段电影质感的短片——比如“暴风雨来临前的废弃太空站&#xff0c;锈迹斑斑的机械臂缓缓抬起&#xff0c;玻璃穹顶外划过流星…

作者头像 李华
网站建设 2026/4/23 12:35:11

JVM调优思路

JVM 调优思路&#xff08;从“瞎调参数”到“可验证的工程方法”&#xff09;这份文档讲的是方法论 常见场景的落地步骤。JVM 调优不是背参数&#xff0c;而是&#xff1a;先测、再定位、再改、再验证。 默认以服务端 Java&#xff08;Spring Boot / 微服务 / 容器&#xff09…

作者头像 李华