【leetcode 力扣刷题】移除链表元素 多种解法

news2025/1/22 19:05:55

移除链表元素的多种解法

  • 203. 移除链表元素
    • 解法①:头节点单独判断
    • 解法②:虚拟头节点
    • 解法③:递归

203. 移除链表元素

题目链接:203.移除链表元素
题目内容:
在这里插入图片描述
理解题意:就是单纯的删除链表中所有值等于给定的val的节点。上一篇博客中介绍了链表的基础操作,在删除链表中节点时,需要注意的是头节点:

  • 如果没有虚拟头节点,那么对头节点的删除需要做不同的处理,head = head->next;
  • 如果有虚拟头节点,那么所有的节点操作都是一致的,将待删除节点的前驱节点和后驱节点连接起来,并释放待删除节点对应的地址空间。

另外,由于链表的定义就是递归的,因此可以考虑使用递归,从最后一个节点开始,判断是否满足待删除条件。

解法①:头节点单独判断

在没有虚拟头节点的情况下,头节点需要单独判断。如果头节点的值就等于val,那么新的头节点head = head->next。然后呢?此时的新head对应节点的值是否等于val呢,如果等于val,那么新的头节点也要删除,删除以后head再次赋值head = head->next。 那么新的head又和给定val相等吗? ——不难发现,判断头节点是否等于val这里应该是循环的,循环直到head->val != va或者head == nullptr才行。
代码如下(C++):

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        //单独判断头节点 【用循环】
        while(head && head->val == val){
            head = head->next;
        }
        ListNode *preNode = NULL, *currNode = head; //preNode是前驱节点,currNode是当前节点
        while(currNode){
            //删除节点
            if(currNode->val == val){
                preNode->next = currNode->next;
                delete currNode;
                currNode = preNode->next;
            }
            //直接后移
            else{
                preNode = currNode;
                currNode = currNode->next;
            }
        }
        return head;
    }
};

如果不一开始就判断头节点,而是在查找值等于val的节点,并删除的过程中,对头节点单独处理的话,怎么知道当前节点是头节点呢? ——删除节点的时候,需要有当前节点currNode的指针,也需要其前驱节点preNode的指针【删除节点的时候,需要将其前驱节点的next指向后驱节点。单向链表通过当前节点可以找到后驱节点,但是不用一个变量存前驱节点的话,是不能通过当前节点直接定位到前驱节点的。】如果当前节点是头节点的话,preNode是NULL,代码如下(C++):

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode* currNode = head;
        ListNode* preNode = NULL;

        while(currNode != nullptr){
            //当前节点和目标val相等,删除
            if(currNode->val == val){
                //删除的是头节点
                if(preNode == NULL){                  
                    head = currNode->next; //新head
                    delete currNode;
                    currNode = head;                
                }
                //删除其他节点
                else{                     
                    preNode->next = currNode->next;
                    delete currNode;
                    currNode = preNode->next;                   
                } 
            }
            //当前节点和目标val不相等,preNode和currNode直接向后移动
            else{
                preNode = currNode;
                currNode = currNode->next;
            }
        }
        return head;
    }
};

解法②:虚拟头节点

添加一个虚拟头节点,那么原链表中所有节点操作都一样,不需要对头节点单独判断,代码如下(C++):

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        ListNode *dummyhead = new ListNode(0);//构造一个虚拟头节点
        dummyhead->next = head; //将head赋值给其next,建立dummyhead和head之间的连接
        ListNode *currNode = head, *preNode = dummyhead; //当前节点、前驱节点
        while(currNode){
            //删除节点
            if(currNode->val == val){
                preNode->next = currNode->next;
                delete currNode;
                currNode = preNode->next;
            }
            else{
                preNode = currNode;
                currNode =currNode->next;
            }
        }
        head = dummyhead->next;
        delete dummyhead; 
        return head;
    }
};

解法③:递归

链表的定义就是递归的,因此可以考虑用递归的方法求解。
思路:①寻找递归终止条件,是head=null;②对于当前节点head,递归调用删除节点的函数,去删除当前节点之后的链表里面满足题意的节点【即removeElements(head->next, val)】,并返回剩下链表删除节点后的头节点;并将removeElements()返回的结果赋值给head->next;③对于当前节点,判断是否满足题意,如果等于val就删除,那么包括当前节点head在内的这一段链表的头节点就是head->next,返回head->next;如果不删除,就直接返回head。

代码如下(C++):

class Solution {
public:
    ListNode* removeElements(ListNode* head, int val) {
        //递归终止条件是head=null,即一直递归调用到最后一个节点的next
        if(head == nullptr)
            return head;
        //当前节点的next指向后半截链表删除节点后返回的头节点
        head->next = removeElements( head->next ,val);
        //判断当前节点是否需要删除
        if(head->val == val)
            return head->next;
        else
            return head;
    }
};

递归求解,不仅要遍历链表各个节点并判断,并且涉及到函数的递归调用,时间和空间开销都比之前解法要大。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/910242.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java【HTTP】什么是 Cookie 和 Session? 如何理解这两种机制的区别和作用?

文章目录 前言一、Cookie1, 什么是 Cookie2, Cookie 从哪里来3, Cookie 到哪里去4, Cookie 有什么用 二、Session1, 什么是 Session2, 理解 Session 三、Cookie 和 Session 的区别总结 前言 各位读者好, 我是小陈, 这是我的个人主页, 希望我的专栏能够帮助到你: 📕 …

并查集 size 的优化(并查集 size 的优化)

目录 并查集 size 的优化 Java 实例代码 UnionFind3.java 文件代码: 并查集 size 的优化 按照上一小节的思路,我们把如下图所示的并查集,进行 union(4,9) 操作。 合并操作后的结构为: 可以发现,这个结构的树的层相对…

juc概述和Lock接口

目录 一、什么是JUC 1、JUC概述 2、进程与线程 3、线程的状态 4、wait/sleep 的区别 5、并发与并行 6、管程 7、用户线程和守护线程 二、Lock接口 1、Synchronized 使用synchronized实现售票案例 使用synchronized实现增减变量操作 2、什么是 Lock 买票例子使用lo…

如何选择 DCDC 降压型开关电源的电感

选择合适的电感是开关电源电路设计的关键之一。本文将帮助您理解电感值和电路性能之间的关系。 降压转换器(buck converter),也称为降压转换器(step-down converter),是一种开关模式稳压器(voltage regulator&#xf…

Mac常见恶意软件再现,办公应用程序潜藏风险如何防范?

Mac电脑正受到臭名昭著的XLoader恶意软件的新变种的攻击,该恶意软件已被重写为在最好的MacBook上本地运行。 虽然XLoader至少从2015年开始出现,但在2021年发现macOS变体之前,它主要用于针对Windows PC。然而,该版本是作为Java程序…

win10系统rust串口通信实现

一、用cargo创建新工程 命令&#xff1a;cargo new comport use std::env; use std::{thread, time}; use serialport::{DataBits, StopBits, Parity, FlowControl}; use std::io::{self, Read, Write}; use std::time::Duration;fn main() -> io::Result<()> {let m…

CSS中的display属性有哪些值?它们的作用?

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ CSS display 属性的不同取值和作用1. block2. inline3. inline-block4. none5. flex6. grid7. table、table-row、table-cell8. list-item9. inline-table、table-caption、table-column 等 ⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#x…

Pika Labs - 用AI工具生成短视频大片

今天我要跟大家分享如何用AI工具1分钟内生成一个短视频大片&#xff0c;效果完全不输影视大V。 只需要用一句话就可以生成视频&#xff0c;或者用一张图就能生成视频&#xff0c;这就是最新推出的AI工具Pika Labs&#xff01;被网友誉为“全球最优秀的文本生成视频AI”。 目前…

SharedPreferences详解及其ANR解决方案

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、商业变现、人工智能等&#xff0c;希望大家多多支持。 目录 一、导读二、概览三、使用四、原理五、存在的问题六、优…

Android动态添加和删除控件/布局

一、引言 最近在研究RecyclerView二级列表的使用方法&#xff0c;需要实现的效果如下。 然后查了一些博客&#xff0c;觉得实现方式太过复杂&#xff0c;而且这种方式也不是特别受推荐&#xff0c;所以请教了别人&#xff0c;得到了一种感觉还不错的实现方式。实现的思路为&…

【LeetCode-经典面试150题-day9]

目录 36.有效的数独 54.螺旋矩阵 48.旋转图像 73.矩阵置零 36.有效的数独 题意&#xff1a; 请你判断一个 9 x 9 的数独是否有效。只需要 根据以下规则 &#xff0c;验证已经填入的数字是否有效即可。 数字 1-9 在每一行只能出现一次。数字 1-9 在每一列只能出现一次。数字 1…

huggingface datasets离线加载文件的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

Sim/circuit10

通过观察可知&#xff0c;在a、b同时为0或1时&#xff0c;state的值改变 state的值可以改变q的输出&#xff0c;1为ab的同或&#xff0c;0为异或 利用assign q进行输出 module top_module (input clk,input a,input b,output q,output state );always(posedge clk)if(a&…

【TypeScript】元组

元组&#xff08;Tuple&#xff09;是 TypeScript 中的一种特殊数据类型&#xff0c;它允许你定义一个固定数量和类型的元素组合。元组可以包含不同类型的数据&#xff0c;每个数据的类型在元组中都是固定的。以下是 TypeScript 中元组的基本用法和特点&#xff1a; // 声明一…

Win系统下安装Linux双系统教程

软件下载 软件&#xff1a;Linux版本&#xff1a;18.0.4语言&#xff1a;简体中文大小&#xff1a;1.82G安装环境&#xff1a;Win11/Win10/Win8/Win7硬件要求&#xff1a;CPU2.0GHz 内存4G(或更高&#xff09;下载通道①丨百度网盘&#xff1a;1.ubuntu18.0.4下载链接&#xf…

量子计算对信息安全的影响:探讨量子计算技术对现有加密方法和信息安全基础设施可能带来的颠覆性影响,以及应对策略

第一章&#xff1a;引言 随着科技的迅猛发展&#xff0c;量子计算作为一项颠覆性的技术正逐渐走入我们的视野。量子计算以其强大的计算能力引发了全球科技界的广泛关注。然而&#xff0c;正如硬币的两面&#xff0c;量子计算技术所带来的不仅仅是计算能力的巨大飞跃&#xff0…

公文校对的艺术:如何确保你的正式文件零错误?

公文是政府和企业中最重要的正式文件之一。一个小小的错误&#xff0c;不仅会影响公文的专业性&#xff0c;甚至可能带来法律和经济后果。因此&#xff0c;如何进行精准的公文校对成为了一门必不可少的技能。接下来&#xff0c;我们将分享一些专业的公文校对技巧&#xff0c;并…

测试框架pytest教程(4)运行测试

运行测试文件 $ pytest -q test_example.py 会运行该文件内test_开头的测试方法 该-q/--quiet标志使输出保持简短 测试类 pytest的测试用例可以不写在类中&#xff0c;但如果写在类中&#xff0c;类名需要是Test开头&#xff0c;非Test开头的类下的test_方法不会被搜集为用…

通过Matlab编程分析微分方程、SS模型、TF模型、ZPK模型的关系

微分方程、SS模型、TF模型、ZPK模型的关系 一、Matlab编程 微分方程、SS模型、TF模型、ZPK模型的关系二、对系统输出进行微分计算三、对系统输出进行积分计算四、总结五、系统的零点与极点的物理意义参考 &#xff1a;[https://www.zhihu.com/question/22031360/answer/3073452…

HCIP---VLAN实验(接入、中继、混杂)

实验要求 PC1/3的接口均为access模式&#xff0c;且属于van2&#xff0c;在同一网段 PC2/4/5/6的IP地址在同一网段&#xff0c;与PC1/3不在同一网段 PC2可以访问4/5/6&#xff0c;PC4不能访问5/6&#xff0c;PC5不能访问PC6 所有PC通过DHCP获取ip地址&#xff0c;PC1/3可以访问…