删除链表中等于给定值 val 的所有节点

news2024/9/19 10:51:27

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

给出链表 1->2->3->3->4->5->3, 和 val = 3, 你需要返回删除3之后的链表:1->2->4->5。
分析思路:这道题的思路,与之前删除链表中重复的结点相似。

因为头结点可能被删除,因此解题的关键在于,创建一个虚结点,将虚结点的next指向原本的头结点。
题目描述:给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val == val 的节点,并返回 新的头节点 。
示例1:

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

提示

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

解题思路:
对于数据结构,我会采用画图进行解题。下面给大家分享一下我的解题思路,请看下图:

这是示例1的输入输出样例,上面的地址是我随便写的,只是为了让大家更好的理解数据结构。这道题还是挺简单的。
只需要让等于val值的那个结点的前一个结点的指针域改成val值结点的指针域就可以了,就可以实现逻辑上的删除,如上图中的第二个结点的指针域改成了第三个结点的指针域,因当我们遍历的时候,它会从第二个结点直接跳到第四个结点,而倒数第二个节点的指针域为null,当遍历到这时就直接停止了,就不会在遍历最后一个结点了。于是便实现了逻辑上的删除。
如下:

这样就可以删掉具有val的结点。只需要加上一些循环条件以及遇到不是val值得结点时的条件就可以了,如下:

        while(cur != null){
            if(cur.val == val){
                prev.next = cur.next;
                cur = cur.next;
             }else{
                 prev = cur;
                 cur = cur.next;
             }
        }

 

以上代码就可以解决很多问题。但是也有隐患,会出现一些特殊情况。
特殊情况1.空链表 2.链表的第一个结点值是val

空链表的情况
空链表的情况处理起来很简单,直接返回null就行了。主要是是否能想到空链表的这个情况,其实对于数据结构的链表题,只要题目中不说明不可能是空链表,就要首先考虑到空链表的情况。

链表的第一个结点值是val
其实这个处理也很简单,上面那段代码可以删除第二个结点以后值是val的结点,那么就只需要在后面判断一下头节点的值是不是val,如果是val就让head = head.next 这样就可以了。
这里有一点要注意一下,判断头节点值是否为val,一定要在循环后面,也就是删除完后面节点的值是val的之后。如果在循环前判断会遇到一些问题,那就是如果第二个结点也是val,那就相当于还是没有删除头节点是val的这种情况。

class Node{
    int val;
    Node next=null;
    Node(int val){
        this.val=val;
    }
    public String toString(){
        return String.format("(%d)", val);
    }
    //头插
public static Node pushFront(Node head,int val){
    //head:原来的第一个结点,val要插入放入值
    Node node=new Node(val); //把val装入一个结点中
    node.next=head;//让原来的 head 成为 node 的下一个结点
    head=node;   //让头成为插入进去的结点
    return head; //返回新的第一个结点
    }
//删除链表中等于定值val的所有结点
  // 方法1:创建一个新链表,遍历原链表把与指定值不相等的移到新链表然后输出
public static  Node removeElements1(Node head,int val){
    if (head == null) {
        return null;
    }
    Node result = null;//创建一个新的链表的引用
    Node last = null;   // 记录目前 result 中的最后一个结点
    Node cur = head;
    while (cur != null) {//这里开始遍历原链表
        if (cur.val == val) {
            cur = cur.next; //碰见与指定值相等的直接跳过此次循环
            continue;
        }
        Node next = cur.next; //记录下一次要遍历的元素
        cur.next = null; //因为不知道这是不是最后一个结点,所以把这个结点的.next=null;
        if (result == null) {
            result = cur; //result指向链表的头
        } else {
            last.next = cur;  //这里是让新链表的倒数第二个结点连接最后一个结点
        }
        last = cur;  //记录最后一个结点
        cur = next;  //cur开始遍历下一个结点
    }
    return result;
}
//方法2:直接在原链表上进行操作,由于删除一个结点必须要知道它的前驱,但是第一个结点往往是没有前驱的,所以我们在前驱问题上有可以有3种不同的实现方法
  //方法2-1:
public static Node removeElements2_1(Node head,int val) {
        if (head == null) {
          return null;
        }
       Node prev =null;  //定义一个引用,它指向cur指向的结点的前一个结点
       Node cur=head;
       while(cur!=null){
           if (cur.val == val) {
               if (cur == head) {  //应对cur是头的情况,直接让head指向下一个结点
                   head = cur.next;
               }
               else {
                   prev.next = cur.next; //prev直接绕过要删除的结点,让它成为垃圾被回收
               }
           } else {
               prev = cur;//如果不是,把prev指向cur指向的结点
           }
           cur=cur.next;
       }
       return head;
    }
    //方法2-2:从第二个结点开始遍历,最后在应对第一个结点的情况
public static Node removeElements2_2(Node head,int val) {
    if (head == null) {
        return null;
    }
    Node prve=head;
    Node cur=head.next;
    while (cur!=null){
        if(cur.val==val){
            prve.next=cur.next;
        }
        else{
            prve=cur;
        }
        cur=cur.next;
    }
    if(head.val==val){   //如果第一个结点就是要删除的结点,那么直接让head指向下一个结点
        head=head.next;
    }
    return head;
    }
    //方法2-3:定义一个没用的结点头插到原链表里,然后打印的时候从第二个结点开始打印
public static Node removeElements2_3(Node head,int val) {
    if (head == null) {
       return null;
    }
    Node tmpHead=new Node(0);
    tmpHead.next=head;//头插到原链表里
    Node prev=tmpHead;
    Node cur=head;
    while(cur!=null){
        if(cur.val==val){
            prev.next=cur.next;
        }else{
            prev=cur;
        }
        cur=cur.next;
    }
    return head;  //这里的head指向的是第二个结点
}
}
public class MyLinkedList {
    public static void main(String[] args){
        Node head=null;  //创建一个空链表
          
        head=Node.pushBack(head,1);
        head=Node.pushBack(head,2);
        head=Node.pushBack(head,6);
        head=Node.pushBack(head,3);
        head=Node.pushBack(head,4);
        head=Node.pushBack(head,5);
        head=Node.pushBack(head,6);
        print(head);
        System.out.println();
        
        Node.removeElements1(head,6);
         print(head)
    }
    //打印函数
    private static void print(Node head){
        System.out.println("打印链表");
        for(Node cur=head;cur!=null;cur=cur.next)
        {
            System.out.print(cur+"-->");
        }
        System.out.print("null");
    }
在这里插入代码片

 

 

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

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

相关文章

腾讯云从业者认证考试考点——云网络产品

文章目录 腾讯云网络产品功能网络产品概述负载均衡&#xff08;Cloud Load Balancer&#xff09;私有网络&#xff08;Virtual Private Cloud&#xff0c;VPC&#xff09;专线接入弹性网卡&#xff08;多网卡热插拔服务&#xff09;NAT网关&#xff08;NAT Gateway&#xff09;…

了解 spring MVC + 使用spring MVC - springboot

前言 本篇介绍什么是spring MVC &#xff0c;如何使用spring MVC&#xff0c;了解如何连接客户端与后端&#xff0c;如何从前端获取各种参数&#xff1b;如有错误&#xff0c;请在评论区指正&#xff0c;让我们一起交流&#xff0c;共同进步&#xff01; 文章目录 前言1. 什么…

RD算法(四)登堂入室 —— 成像完成

SAR成像专栏目录_lightninghenry的博客-CSDN博客https://lightning.blog.csdn.net/article/details/122393577?spm=1001.2014.3001.5502先放RD算法最终的成像结果: 经简单的地距投影后为(地距投影的内容在后面的几何校正章节中讲解): 温哥华这地形还真像是一张怪兽的巨嘴呀…

商城-学习整理-基础-商品服务API-三级分类(五)

目录 一、启动之前创建的环境1、启动虚拟机2、启动mysql3、启动redis4、启动nacos5、导入三级分类测试数据 二、开发商品服务三级分类列表1、后台模块1&#xff09;书写商品三级分类表后台业务逻辑 2、前端模块1&#xff09;启动renren-fast前后端项目2&#xff09;在系统管理-…

js逆向 - X-Zse-96分析(JsRPC实战)

本文仅供学习交流&#xff0c;只提供关键思路不会给出完整代码&#xff0c;严禁用于非法用途&#xff0c;若有侵权请联系我删除&#xff01; 目标网站&#xff1a;aHR0cHM6Ly93d3cuemhpaHUuY29tLw 目标接口&#xff1a;aHR0cHM6Ly93d3cuemhpaHUuY29tL2FwaS92NC9zZWFyY2hfdjM …

卸载 Postman!一款 IDEA 神级插件,更便捷、高效...

Restful Fast Request 是 IDEA 版 Postman&#xff0c;它是一个强大的 restful api 工具包插件&#xff0c;可以根据已有的方法帮助您快速生成 url 和 params。 Restful Fast Request API 调试工具 API 管理工具 API 搜索工具。 它有一个漂亮的界面来完成请求、检查服务器…

Django智能养殖场管理系统

目 录 摘要 1 绪论 1.1研究背景与意义 1.2研究内容 1.3论文结构与章节安排 2 智能养殖场管理系统分析 2.1 可行性分析 2.2 系统业务流程分析 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析 2.5本章小结 3 智能养殖场管理系统总体设计 …

数据分析 VS 数据可视化:决战时刻

数据分析和数据可视化是数据科学领域中两个重要的组成部分&#xff0c;很多人不明白两者之间的关系&#xff0c;会误认为是一个东西&#xff0c;其实不然。本文就带大家简单了解一下它们的区别与联系吧&#xff01; 数据分析是指通过收集、处理和解释数据来获取有关特定问题或…

深度学习入门 ---- 张量(Tensor)

文章目录 张量张量在深度学习领域的定义张量的基本属性使用PyTorch安装PyTorch查看安装版本 创建张量常用函数四种创建张量的方式和区别 四则运算 张量 张量在深度学习领域的定义 张量&#xff08;tensor&#xff09;是多维数组&#xff0c;目的是把向量、矩阵推向更高的维度。…

Docker中CMD和ENTRYPOINT的区别

目录 一.CMD二.ENTRYPOINT三.总结 一.CMD CMD指令为启动的容器指定默认要运行的程序&#xff0c;程序运行结束&#xff0c;容器也就结束。CMD 指令指定的程序可被 docker run 命令行参数中指定要运行的程序所覆盖。类似于 RUN 指令&#xff0c;用于运行程序&#xff0c;但二者…

Java:JDK8之后新的时间(推荐使用) LocalDate、LocalTime、LocalDataTime 的相关API

更新后的时间分类 LocalDate 的相关API //0、获取本地日期对象 LocalDate ld LocalDate.now();//年月日 system.out.println(ld) ;//1、获取日期对象中的信息 int year ld.getYear();//年 int month ld.getMonthValue(); //月(1-12) int day ld.getDayOfMonth();//日 int d…

不用下软件,三维模型就可以实现在线浏览了!

对三维有一些了解的朋友都知道&#xff0c;目前很多三维模型网站基本都是以2D形式展示的&#xff0c;用户只能通过截取不同角度的视图来查看模型。如果用户想要实现模型动态预览&#xff0c;只能下载其模型文件&#xff0c;并在对应的三维软件中打开。现在&#xff0c;不需要下…

springBoot多数据源使用tdengine(3.0.7.1)+MySQL+mybatisPlus+druid连接池

一、安装部署 1、我这里使用的 3.0.7.1版本&#xff0c;因为我看3.x版本已经发布了一年了&#xff0c;增加了很多新的功能&#xff0c;而且3.x官方推荐&#xff0c;对于2.x的版本&#xff0c;官网都已经推荐进行升级到3.x&#xff0c;所以考虑到项目以后的发展&#xff0c;决定…

EMC VxRail E560 vusb0端口不断down和up问题

客户的一台VxRail E560报vusb0网卡故障&#xff0c;如下所示&#xff1a; 该端口是VxRail ESXi与iDRAC内部通信的网口&#xff0c;并没有连接外部网络。 在vobd.log日志中&#xff0c;可以看到vusb0不断在down和up的信息&#xff1a; 由于节点在2023年4月份进行了重装&am…

每日一题8.2 2536

2536. 子矩阵元素加 1 给你一个正整数 n &#xff0c;表示最初有一个 n x n 、下标从 0 开始的整数矩阵 mat &#xff0c;矩阵中填满了 0 。 另给你一个二维整数数组 query 。针对每个查询 query[i] [row1i, col1i, row2i, col2i] &#xff0c;请你执行下述操作&#xff1a;…

超出背包限制--潜水员

本题和之前的二维背包差不多&#xff0c;但是它可以做到超出体积的限制&#xff0c;本质上我们的状态转移方程其实和之前的差不多&#xff0c;f[i][j]min(f[i-1][j],f[i-1][j-v]w),超出的体积应该如何去计算&#xff0c;我们可以利用负数去计算&#xff0c;也就是i-v1<0||j-…

如何在烟草行业运用IPD?

从当前的世界烟草行业来看&#xff0c;烟草经济的发展十分迅速&#xff0c;中国是烟草生产与消费第一大国&#xff0c;每年由我国生产与出售的烟草远销世界各地。与此同时&#xff0c;中国烟草行业的集中度越来越高&#xff0c;企业的数量与规模稳步上升&#xff0c;行业迈向规…

Flink - sink算子

水善利万物而不争&#xff0c;处众人之所恶&#xff0c;故几于道&#x1f4a6; 文章目录 1. Kafka_Sink 2. Kafka_Sink - 自定义序列化器 3. Redis_Sink_String 4. Redis_Sink_list 5. Redis_Sink_set 6. Redis_Sink_hash 7. 有界流数据写入到ES 8. 无界流数据写入到ES 9. 自定…

nodejs环境部署

1.前言 前端的打包必须要使用nodejs环境 2.部署 下载安装包&#xff0c;在下面的路径中找到适合自己项目合适的版本在下载&#xff0c;可以问一下前端开发人员 路径&#xff1a;Index of /dist/ 我这里是下载了14.21.3版本的nodejs环境,上传到需要部署的主机中 mkdir /opt…

制砖机系统比例控制阀放大器

制砖机系统是一种生产砖块的机器设备系统。该系统由多个部分组成&#xff0c;包括压力系统、模具和振动系统、烘干和烧制系统等。压力系统是制砖机的主要组成部分之一&#xff0c;它通过压力将原料压缩成一定形状和尺寸的块状&#xff0c;然后经过烘干和烧制等步骤&#xff0c;…