[LeetCode]day9 203.移除链表元素

203. 移除链表元素 - 力扣(LeetCode)

题目描述

给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val == val 的节点,并返回 新的头节点

示例 1:

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

示例 2:

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

示例 3:

复制代码
输入:head = [7,7,7,7], val = 7
输出:[]

提示:

  • 列表中的节点数目在范围 [0, 104]
  • 1 <= Node.val <= 50
  • 0 <= val <= 50

解题

数据结构里早就学过怎么删除链表元素了,原本以为可以手拿把掐,没想到一写就出各种细节上的问题!!

这道题唯一的难点就是在于怎么处理返回这个头结点,题目中的这个头结点是带有数据的,而不是指向第一个有效结点的结点 ,所以是有可能为待删除元素的,而删除这个头结点和删除其他节点的操作有所不同。

解法一:使用原来的链表来进行移除节点操作

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* removeElements(ListNode* head, int val) {
        ListNode*p;
        if(head==NULL)return NULL;
        //处理头结点为待删结点的情况
        while(head != nullptr&&head->val==val){ //注意这个条件判断的前后顺序
           p=head;
           head=head->next;
           delete p;  //c/c++要记得释放内存
        }
        p=head; 
        while(p&&p->next){ //要这两个都不为空才行
            if(p->next->val==val){
                ListNode*q=p->next;
                p->next=p->next->next;
                delete q;
            }else{
                p=p->next; //这里不是每次都要执行的
            }
        
        }
        return head;
    }
};

有几个点特别容易错..

1.while(head != nullptr&&head->val==val) 两个判断的前后顺序千万不能反..一定要先判断是否为空,再判断是否等于val. 因为当head为空时先判断head->val==val会造成访问空指针的错误!

2.使用c/c++要记得释放内存!

3.while(p&&p->next) 两个条件都要满足 因为后面的删除操作会访问p->next;

4.p=p->next不是每次循环都要执行的,只有不删除元素时才执行,如果删除了元素就不执行(可以画个图进行理解)


解法二:引入虚拟头结点

如果我们引入一个虚拟的头结点,那删除头结点和其他结点的操作就统一了。

cpp 复制代码
class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode*dummy=new ListNode(0),*q;
        dummy->next=head;
        q=dummy;
        while(q&&q->next){
            if(q->next->val==val){
                ListNode*p=q->next;
                q->next=q->next->next;
                delete p;
            }else{
                q=q->next;
            }
        }
        return dummy->next;
    }
};

这里有一个易错点就是要记得为虚拟头结点申请空间

相关推荐
代码小将1 小时前
Leetcode209做题笔记
java·笔记·算法
Musennn2 小时前
leetcode 15.三数之和 思路分析
算法·leetcode·职场和发展
CM莫问5 小时前
<论文>(微软)避免推荐域外物品:基于LLM的受限生成式推荐
人工智能·算法·大模型·推荐算法·受限生成
康谋自动驾驶6 小时前
康谋分享 | 自动驾驶仿真进入“标准时代”:aiSim全面对接ASAM OpenX
人工智能·科技·算法·机器学习·自动驾驶·汽车
C++ 老炮儿的技术栈7 小时前
什么是函数重载?为什么 C 不支持函数重载,而 C++能支持函数重载?
c语言·开发语言·c++·qt·算法
yychen_java7 小时前
R-tree详解
java·算法·r-tree
MarkHard1238 小时前
Leetcode (力扣)做题记录 hot100(62,64,287,108)
算法·leetcode·职场和发展
王RuaRua8 小时前
[数据结构]5. 栈-Stack
linux·数据结构·数据库·链表
一只鱼^_8 小时前
牛客练习赛138(首篇万字题解???)
数据结构·c++·算法·贪心算法·动态规划·广度优先·图搜索算法
一只码代码的章鱼8 小时前
Spring的 @Validate注解详细分析
前端·spring boot·算法