算法-链表实战【删除链表的倒数第 N 个结点】中等

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

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

示例 1:

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

示例 2:

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

示例 3:

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

提示:

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

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

解答

核心思路:​快慢双指针法

--->使用两个指针(快慢指针),让快指针先走 N 步,然后两个指针同步移动。当快指针到达尾部时,慢指针正好指向倒数第 N 个节点的前驱节点,完成删除操作。

ini 复制代码
class ListNode {
    int val;
    ListNode next;
    ListNode() {}
    ListNode(int val) { this.val = val; }
    ListNode(int val, ListNode next) { this.val = val; this.next = next; }
}

class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        // 1. 创建哑节点(哨兵节点),用于处理头节点删除情况
        ListNode dummy = new ListNode(0);
        dummy.next = head;
        
        // 2. 初始化快慢指针,都从哑节点开始
        ListNode fast = dummy;
        ListNode slow = dummy;
        
        // 3. 快指针先走 N 步
        for (int i = 0; i <= n; i++) {
            fast = fast.next;
        }
        
        // 4. 同步移动快慢指针
        while (fast != null) {
            fast = fast.next;
            slow = slow.next;
        }
        
        // 5. 删除倒数第 N 个节点
        slow.next = slow.next.next;
        
        // 6. 返回新头节点(始终是 dummy.next)
        return dummy.next;
    }
}

示例说明(以输入head = [1,2,3,4,5], n=2为例)

  1. 创建哑节点dummy(0) -> 1 -> 2 -> 3 -> 4 -> 5
  2. 初始化指针fastslow都指向dummy(0)
  3. 快指针移动n+1=3步
    • 第一次:fast=1
    • 第二次:fast=2
    • 第三次:fast=3
  4. 同时移动
    • 快指针从3开始移动到尾部(4,5,null),慢指针从0开始移动到1,2,3。当快指针为null时,慢指针在3的位置
  5. 删除节点slow(3)的下一个节点是4(即倒数第2个节点),修改其next指向5,链表变为dummy(0)->1->2->3->5
  6. 返回dummy.next=1,即返回[1,2,3,5]

完整算法步骤

  1. 创建哑节点
    • 新建一个哑节点dummy,并将其next指向链表的头节点。
  2. 初始化两个指针
    • fast(快指针)和slow(慢指针),都指向哑节点。
  3. 快指针先移动N+1步
    • 这样快指针和慢指针之间就保持了N+1个节点的距离。当快指针移动到链表末尾时,慢指针刚好指向待删除节点的前驱节点。
  4. 同时移动快指针和慢指针
    • 当快指针不为null时,快慢指针同时向后移动一步,直到快指针指向null(即快指针到达链表尾部)。
  5. 删除节点
    • 此时慢指针slow指向待删除节点的前一个节点,将slow.next指向slow.next.next,就删除了倒数第N个节点。
  6. 返回新链表的头节点
    • 返回哑节点dummynext,即新链表的头节点。
相关推荐
LYFlied3 小时前
【每日算法】LeetCode 153. 寻找旋转排序数组中的最小值
数据结构·算法·leetcode·面试·职场和发展
jianfeng_zhu6 小时前
整数数组匹配
数据结构·c++·算法
yueqingll6 小时前
032、数据结构之代码时间复杂度和空间复杂度的判断:从入门到实战
数据结构
罗湖老棍子9 小时前
最小函数值(minval)(信息学奥赛一本通- P1370)
数据结构·c++·算法··优先队列·
LYFlied9 小时前
【每日算法】LeetCode 208. 实现 Trie (前缀树)
数据结构·算法·leetcode·面试·职场和发展
AI科技星10 小时前
统一场论框架下万有引力常数的量子几何涌现与光速关联
数据结构·人工智能·算法·机器学习·重构
仰泳的熊猫10 小时前
1109 Group Photo
数据结构·c++·算法·pat考试
2401_8414956411 小时前
【数据结构】最短路径的求解
数据结构·动态规划·贪心·ipython·最短路径·迪杰斯特拉算法·弗洛伊德算法
tgethe11 小时前
Java 数组(Array)笔记:从语法到 JVM 内核
java·数据结构
客梦11 小时前
数据结构-单链表
数据结构