算法-链表实战【删除链表的倒数第 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,即新链表的头节点。
相关推荐
多米Domi0112 小时前
0x3f 第49天 面向实习的八股背诵第六天 过了一遍JVM的知识点,看了相关视频讲解JVM内存,垃圾清理,买了plus,稍微看了点确定一下方向
jvm·数据结构·python·算法·leetcode
L_090711 小时前
【C++】高阶数据结构 -- 红黑树
数据结构·c++
划破黑暗的第一缕曙光15 小时前
[数据结构]:5.二叉树链式结构的实现1
数据结构
青桔柠薯片15 小时前
数据结构:单向链表,顺序栈和链式栈
数据结构·链表
XiaoFan01215 小时前
将有向工作流图转为结构树的实现
java·数据结构·决策树
睡一觉就好了。15 小时前
快速排序——霍尔排序,前后指针排序,非递归排序
数据结构·算法·排序算法
齐落山大勇16 小时前
数据结构——单链表
数据结构
皮皮哎哟16 小时前
深入浅出双向链表与Linux内核链表 附数组链表核心区别解析
c语言·数据结构·内核链表·双向链表·循环链表·数组和链表的区别
wWYy.17 小时前
指针与引用区别
数据结构
历程里程碑17 小时前
Linux 17 程序地址空间
linux·运维·服务器·开发语言·数据结构·笔记·排序算法