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

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

中等

提示

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

示例 1:

复制代码
输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]

示例 2:

复制代码
输入:head = [1], n = 1
输出:[]

示例 3:

复制代码
输入:head = [1,2], n = 1
输出:[1]

提示:

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

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

📝 核心笔记:删除链表的倒数第 N 个节点

1. 核心思想 (一句话总结)

"拿着一把长为 N+1 的尺子往后滑。"

让 right 指针先走N 步,保持 left 和 right 之间固定的距离。当 right 走到链表尽头时,left 恰好停在待删除节点的前一个位置(Pre-node)。

2. 为什么要用哨兵 (Dummy Node)?
  • 统一逻辑 :如果要删除的是头节点 (倒数第 Length 个),没有 Dummy 的话需要单独写 if 判断。
  • 有了 Dummy,left 永远从 Dummy 开始,删除头节点就变成了"删除 Dummy 的下一个",逻辑完全一致。
3. 逻辑图解 (尺子平移)

假设删除倒数第 2 个 (n=2):

  1. 拉开间距right 先出发,走了n 步。
  2. 同步平移leftright 同时走,直到 right 到底。
  3. 定位 :此时 right 在最后,left 在待删除节点的前面 (Pre)。
4. 代码回忆清单 (带注释版)
复制代码
// 题目:LC 19. 删除链表的倒数第 N 个结点
class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        // 1. 哨兵节点 (关键!处理删除头节点的情况)
        ListNode dummy = new ListNode(0, head);
        ListNode left = dummy;
        ListNode right = dummy; // 两人都从哨兵出发

        // 2. 拉开间距 (Right 先走 n 步)
        // 目标:让 right 和 left 之间隔着 n 个节点
        while (n-- > 0) {
            right = right.next;
        }

        // 3. 同步移动 (直到 Right 走到最后一个节点)
        // 注意:这里是 right.next != null,让 right 停在尾节点
        while (right.next != null) {
            left = left.next;
            right = right.next;
        }

        // 此时:
        // right 在 尾巴
        // left 在 待删除节点的前驱 (Pre)
        
        // 4. 执行删除
        left.next = left.next.next;

        return dummy.next; // 返回哨兵的后面
    }
}

⚡ 快速复习 CheckList (易错点)

  • \] **哨兵节点加了吗?**

    • 必须加。ListNode dummy = new ListNode(0, head); 否则删头节点会报错。
  • \] **Right 先走几步?**

    • n 步。
  • \] **循环结束条件?**

    • while (right.next != null)。我们希望 right 停在最后一个有效节点,这样 left 刚好在待删节点的前一个。
  • \] **最后返回谁?**

    • return dummy.next。千万别返回 head,因为 head 可能已经被删掉了。

🧠 场景联想

想象你手里拿了一根长棍子(长度 n)。

  • 你把棍子的一头(Right)顶在链表的最末端。
  • 棍子的另一头(Left)自然就指在了倒数第 n 个位置的前面。
    • 直接切断,完事。
相关推荐
WeiXiao_Hyy7 小时前
成为 Top 1% 的工程师
java·开发语言·javascript·经验分享·后端
苏渡苇7 小时前
优雅应对异常,从“try-catch堆砌”到“设计驱动”
java·后端·设计模式·学习方法·责任链模式
团子的二进制世界7 小时前
G1垃圾收集器是如何工作的?
java·jvm·算法
吃杠碰小鸡7 小时前
高中数学-数列-导数证明
前端·数学·算法
故事不长丨7 小时前
C#线程同步:lock、Monitor、Mutex原理+用法+实战全解析
开发语言·算法·c#
long3167 小时前
Aho-Corasick 模式搜索算法
java·数据结构·spring boot·后端·算法·排序算法
近津薪荼7 小时前
dfs专题4——二叉树的深搜(验证二叉搜索树)
c++·学习·算法·深度优先
熊文豪7 小时前
探索CANN ops-nn:高性能哈希算子技术解读
算法·哈希算法·cann
熊猫_豆豆8 小时前
YOLOP车道检测
人工智能·python·算法
rannn_1118 小时前
【苍穹外卖|Day4】套餐页面开发(新增套餐、分页查询、删除套餐、修改套餐、起售停售)
java·spring boot·后端·学习