力扣HOT10(29) 删除链表的倒数第 N 个结点

一、先搞懂链表删除操作的核心本质

⚠️ 所有链表删除题的第一原则:要删除一个节点,必须先找到它的前驱节点!

比如要删除节点 4,你不能直接操作 4,必须先找到它的前一个节点 3,然后执行:

3->next = 3->next->next;

这样就把 4 从链表中删掉了。

这就是为什么这道题我们的目标不是找到倒数第 N 个节点,而是找到倒数第 N 个节点的前驱节点

二、为什么必须用虚拟头节点?

如果要删除的节点是头节点 (比如示例 2:[1],删除倒数第 1 个),头节点没有前驱节点,需要单独写代码处理,非常容易出错。

用虚拟头节点的好处:

  • 给头节点也造一个前驱,所有节点的删除逻辑完全统一,不用再单独处理头节点的情况。

三、方法一:计算链表长度

核心思路

  1. 先遍历一遍链表,计算出链表的总长度L
  2. 倒数第 N 个节点,就是正数第L - N + 1个节点;
  3. 我们需要找到它的前驱,也就是正数第L - N个节点;
  4. 执行删除操作。

方法二:栈

核心思路

  1. 把所有节点(包括虚拟头)依次入栈;
  2. 弹出 n 个节点,此时栈顶的节点就是倒数第 N 个节点的前驱;
  3. 执行删除操作。

方法三:双指针法(面试首选!进阶要求,O (L) 时间,O (1) 空间)

1.核心思想(一句话记死)

让快指针先走 n 步,然后快慢指针一起走,当快指针走到链表末尾时,慢指针正好指向倒数第 n 个节点的前驱。

2. 为什么这样是对的?

  • 快指针先走 n 步,此时快慢指针之间的距离正好是 n;
  • 然后两个指针以相同速度一起走,距离保持不变;
  • 当快指针走到nullptr时,慢指针和末尾的距离正好是 n,也就是慢指针在倒数第 n 个节点的前一个位置。

⚠️ 关键细节:慢指针必须从虚拟头节点开始,快指针从原头节点开始

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* removeNthFromEnd(ListNode* head, int n) {



        
        //双指针法
        //两个指针

        //虚拟头
        ListNode *dhead = new ListNode(-1);
        dhead->next = head;

        ListNode *p1 = dhead;
        ListNode *p2 = head;


        //p2先走n步
        int step = 0;//记录走了多少步
        while(step<n){
            p2 = p2->next;
            step++;
        }

        //p1 p2同时走 直到p2走到头 p1的位置就是要删除的节点的前驱节点
        while(p2!=nullptr){

            p1 = p1->next;
            p2 = p2->next;

        }
        
        p1->next = p1->next->next;
       // while(p1!=nullptr){

       //     p1 = p1->next;
       // }

        return dhead->next;
    }
};
相关推荐
yuan199976 分钟前
欧拉梁静力与屈曲计算的 MATLAB 实现(有限差分法 + 解析解)
开发语言·算法·matlab
玖玥拾30 分钟前
C/C++ 数据结构(六)链表迭代器与底层
c语言·数据结构·c++·链表·stl库
汉克老师1 小时前
GESP7级C++考试语法知识(二、指数函数(3、综合练习)
c++·算法·数学建模·指数函数·gesp7级·复利
林间码客2 小时前
04 ROC曲线与AUC:从零开始手动计算
大数据·人工智能·算法
Irissgwe2 小时前
map/set/multimap/multiset 的底层逻辑与实现
数据结构·c++·算法·二叉树·stl·c·红黑树
圣保罗的大教堂2 小时前
leetcode 2130. 链表最大孪生和 中等
leetcode
IronMurphy2 小时前
【算法五十八】23. 合并 K 个升序链表
数据结构·算法·链表
思茂信息2 小时前
CST软件基于液态金属开关的方向图可重构天线
服务器·算法·重构·cst·仿真软件·电磁仿真
月疯2 小时前
PPG研究中暑的算法记录
算法
春日见2 小时前
vscode的AI编程插件推荐:
大数据·ide·vscode·算法·机器学习·编辑器·ai编程