删除链表元素相关的练习

news2025/1/13 7:33:36

目录

一、移除链表元素

二、删除排序链表中的重复元素

三、删除排序链表中的重复元素 ||

四、删除链表的倒数第 N 个结点



一、移除链表元素

给你一个链表的头节点 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, 10^4]

  • 1 <= Node.val <= 50

  • 0 <= val <= 50

代码实现一

struct ListNode* removeElements(struct ListNode* head, int val)
{
    struct ListNode* pre = NULL;
    struct ListNode* cur = head;
    while (cur != NULL)
    {
        if (cur->val == val)
        {
            if (pre == NULL)  // 当删除的是首元结点时,要做特殊处理,避免访问空指针
            {
                head = cur->next;
                free(cur);
                cur = head;  // 切不可写成 cur = cur->next
            }
            else
            {
                pre->next = cur->next;
                free(cur);
                cur = pre->next;  // 切不可写成 cur = cur->next
            }
        }
        else
        {
            pre = cur;
            cur = cur->next;
        }
    }
    return head;
}

free(cur); cur = cur->next 是一种类似刻舟求剑的典型的错误写法

代码实现二

struct ListNode* removeElements(struct ListNode* head, int val)
{
    struct ListNode* newhead = NULL;
    struct ListNode* tail = NULL;
    struct ListNode* cur = head;
    while (cur != NULL)
    {
        if (cur->val != val)
        {
            if (newhead == NULL)  // 如果新链表为空
            {
                newhead = cur;
                tail = cur;
            }
            else
            {
                tail->next = cur;
                tail = cur;
            }
            cur = cur->next;
        }
        else
        {
            struct ListNode* tmp = cur;
            cur = cur->next;
            free(tmp);
        }
    }
    if (tail != NULL)  // 避免对空指针的解引用
    {
        tail->next = NULL;  
    }
    return newhead;
}

将值不是 val 的结点尾插到新链表中由于尾插需要从头指针出发顺链找到尾结点,时间复杂度高,因此可以用一个尾指针 tail 来记录并更新尾结点的位置

图解示例一

 


二、删除排序链表中的重复元素

给定一个已排序的链表的头 head删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表

示例 1

输入:head = [1,1,2]

输出:[1,2] 

示例 2

输入:head = [1,1,2,3,3]

输出:[1,2,3] 

提示

  • 链表中节点数目在范围 [0, 300]

  • -100 <= Node.val <= 100

  • 题目数据保证链表已经按升序排列

代码实现

struct ListNode* deleteDuplicates(struct ListNode* head) 
{
    if (head == NULL)  // 判断是否为空表
    {
        return NULL;
    }
    struct ListNode* tail = head;
    struct ListNode* cur = head->next;
    while (cur != NULL)
    {
        if (cur->val != tail->val)
        {
            // 尾插
            tail->next = cur;
            tail = cur;
            cur = cur->next;
        }
        else
        {
            // 删除重复的元素
            struct ListNode* tmp = cur;
            cur = cur->next;
            free(tmp);
        }
    }
    tail->next = NULL;
    return head;
}


三、删除排序链表中的重复元素 ||

给定一个已排序的链表的头 head删除原始链表中所有重复数字的节点,只留下不同的数字 。返回 已排序的链表

示例 1

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

输出:[1,2,5]

示例 2

 输入:head = [1,1,1,2,3]

输出:[2,3]

提示

  • 链表中节点数目在范围 [0, 300]

  • -100 <= Node.val <= 100

  • 题目数据保证链表已经按升序排列

代码实现

struct ListNode* deleteDuplicates(struct ListNode* head)
{
    struct ListNode* newhead = NULL;
    struct ListNode* tail = NULL;
    struct ListNode* cur = head;
    while (cur != NULL)
    {
        // 找到第一个值不同于 cur 的结点(也可能为空)
        struct ListNode* after = cur->next;
        while (after != NULL && after->val == cur->val)
        {
            after = after->next;
        }
        // 判断 cur 是否属于有重复数字的结点
        if (cur->next == after)  // 不属于
        {
            // 尾插
            if (newhead == NULL)  // 如果新链表为空
            {
                newhead = tail = cur;
            }
            else
            {
                tail->next = cur;
                tail = cur;
            }
            cur = cur->next;
        }
        else  // 属于
        {
            while (cur != after)
            {
                struct ListNode* tmp = cur;
                cur = cur->next;
                free(tmp);
            }
        }
    }
    if (tail != NULL)
    {
        tail->next = NULL;
    }
    return newhead;
}


四、删除链表的倒数第 N 个结点

给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。

示例 1

 输入:head = [1,2,3,4,5], n = 2

输出:[1,2,3,5]

示例 2

输入:head = [1], n = 1

输出:[]

示例 3

输入:head = [1,2], n = 1

输出:[1]

提示

  • 链表中结点的数目为 sz

  • 1 <= sz <= 30

  • 0 <= Node.val <= 100

  • 1 <= n <= sz

代码实现一

struct ListNode* removeNthFromEnd(struct ListNode* head, int n) 
{
    struct ListNode* pre_slow = NULL;
    struct ListNode* slow = head;
    struct ListNode* fast = head;
    while (--n)
    {
        fast = fast->next; 
        // 由于 1 <= n <= sz,所以 fast 不可能走到空
    }
    while (fast->next != NULL)
    {
        pre_slow = slow;
        slow = slow->next;
        fast = fast->next;
    }
    if (pre_slow == NULL)
    {
        head = slow->next;
        free(slow);
    }
    else
    {
        pre_slow->next = slow->next;
        free(slow);
    }
    return head;
}

相关练习:链表的中间结点、链表中倒数第 k 个结点

代码实现二

struct ListNode* removeNthFromEnd(struct ListNode* head, int n) 
{
    // 设置一个哨兵位的头结点
    struct ListNode* guard = (struct ListNode*)malloc(sizeof(struct ListNode));
    guard->next = head;
    // 计算链表的长度(包括头结点)
    int len = 0;
    struct ListNode* cur = guard;
    while (cur != NULL)
    {
        ++len;
        cur = cur->next;
    }
    // 找到倒数第 n + 1 个结点(可能是 guard)
    struct ListNode* pre = guard;
    for (int i = 0; i < len - n - 1; ++i)
    {
        pre = pre->next;
    }
    // 删除倒数第 n 个结点
    struct ListNode* tmp = pre->next;
    pre->next = tmp->next;
    free(tmp);
    // 删除哨兵位的头结点并返回 head
    head = guard->next;
    free(guard);
    return head;
}

设置哨兵位的头结点的好处是便于首元结点的处理

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

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

相关文章

python字典和集合——笔记

一、介绍 1、泛映射类型 collections.abc模块中有Mapping和MutableMapping这两个抽象基类&#xff0c;它们的作用是为dict和其他类似的类型定义形式接口&#xff08;在Python 2.6到Python 3.2的版本中&#xff0c;这些类还不属于collections.abc模块&#xff0c;而是隶属于coll…

【震撼发布】《致敬未来的攻城狮计划》| 文末赠书3本

《致敬未来的攻城狮计划》—— 文末有福利 摘要&#xff1a; 一个崭新的计划&#xff0c;寻找那群有志于向嵌入式发展的未来工程师&#xff01; 文章目录1 活动计划初衷2 活动计划形式3 活动计划收获4 活动计划要求5 活动计划时间6 活动计划致谢7 活动计划特别说明8 温馨提示9 …

Kerberos 域委派攻击之非约束性委派

CSDN文章自动迁移自博客在Windows 2000 Server 首次发布 Active Directory 时&#xff0c;Microsoft 必须提供一种简单的机制来支持用户通过 Kerberos 向 Web Server 进行身份验证并需要代表该用户更新后端数据库服务器上的记录的方案。这通常称为“Kerberos 双跳问题”&#x…

零入门kubernetes网络实战-20->golang编程syscall操作tun设备介绍

《零入门kubernetes网络实战》视频专栏地址 https://www.ixigua.com/7193641905282875942 本篇文章视频地址(稍后上传) 本篇文章主要是使用golang自带的syscall包来创建tun类型的虚拟网络设备。 注意&#xff1a; 目前只能使用syscall包来创建tun类型的虚拟设备。 tun虚拟网…

【RockerMQ】003-Windows 安装 RocketMQ

【RockerMQ】003-Windows 安装 RocketMQ 一、准备工作 1、环境要求 64位JDK 1.8;Maven 3.2.x;64位操作系统系统&#xff0c;本文档在Windows上安装 2、下载解压 下载地址 https://archive.apache.org/dist/rocketmq/5.1.0/ 下载目标 解压 到不含中文路径的目录下 环境变…

Python进阶-----面对对象7.0(细谈__new__方法和__init__方法)

目录 前言&#xff1a; __init__方法 __new__方法&#xff08;重点!&#xff09; 1.__new__方法的调用过程 2.重写__new__方法 3.__new__方法不同返回值的情况 3.单例模式 前言&#xff1a; 上一期初步介绍了__new__()方法&#xff0c;但是实际上这个方法还有非常多的内…

操作系统——18.进程互斥的软件实现方法

这篇文章我们来讲一下进程互斥的软件实现方法 目录 1.概述 2.单标志法 3.双标志检查法 4.双标志后检查法 5.Perterson算法 6.小结 1.概述 首先&#xff0c;我们来看一下这节的大体框架 学习提示: 理解各个算法的思想、原理结合上小节学习的“实现互斥的四个逻辑部分”…

通用业务平台设计(五):预警平台建设

前言 在上家公司&#xff0c;随着业务的不断拓展(从支持单个国家单个主体演变成支持多个国家多个主体)&#xff0c;对预警的诉求越来越紧迫&#xff1b;如何保障业务的稳定性那&#xff1f;预警可以帮我们提前甄别风险&#xff0c;从而让我们可以在风险来临前将其消灭&#xff…

深度学习笔记:dropout和调优超参数方法

1 Dropout Dropout是一个常用于深度学习的减轻过拟合的方法。该方法在每一轮训练中随机删除部分隐藏层神经元。被删除的神经元不会进行正向或反向信号传递。在测试阶段所有神经元都会传递信号&#xff0c;但对各个神经元的输出要乘以训练时删除比例。 Dropout实现程序&#xf…

毕业设计 基于STM32单片机无线ZIGBEE智能大棚土壤湿度光照检测

基于STM32单片机无线ZIGBEE智能大棚土壤湿度光照检测1、项目简介1.1 系统构成1.2 系统功能2、部分电路设计2.1 STM32F103C8T6核心系统电路设计2.2 光敏采集电路设计2.3 温度采集电路设计3、部分代码展示3.1 读取DS18B20温度值3.2 定时器初始化1、项目简介 选题指导&#xff0c…

Learning Typescript and React in ts

目录 Basic typescript What is typescript? Configuring the TypeScript Compiler Fundamental build in types TypeScript Simple Types TypeScript Special Types Type: unknown Type: never Type: undefined & null Arrays Tuple Enums functions Ob…

Java集合专题

文章目录框架体系CollectionListArrayListLinkedListVectorSetHashSetLinkedHashSetTreeSetMapHashMapHashtableLinkedHashMapTreeMapPropertiesCollections框架体系 1、集合主要分了两组&#xff08;单列集合&#xff0c;双列集合&#xff09; 2、Collection接口有两个重要的子…

2.SpringSecurity认证

2.1登录校验流程 2.2认证原理 *源码流程: *自定义认证流程: *校验流程: *认证和校验连接: 2.3思路分析 *登录:

SQLI-Labs通关(2)5-7关

跟之前一样首先传参&#xff0c;然后查看注入点以及闭合 用and 11 and 12都没问题&#xff0c;接下来测试单引号 利用 and 12的时候会报错 利用order by来判断列数 得出一共三列 接下来就是联合查询 但是这个并不会回显 那么就利用盲注或者报错注入 在这里我们利用报错来测…

Vue3的生命周期函数

文章目录&#x1f31f; 写在前面&#x1f31f; 生命周期钩子函数&#x1f31f; 组合式API生命周期&#x1f31f; 写在最后&#x1f31f; 写在前面 专栏介绍&#xff1a; 凉哥作为 Vue 的忠实 粉丝输出过大量的 Vue 文章&#xff0c;应粉丝要求开始更新 Vue3 的相关技术文章&am…

OPPO 数据恢复:如何从 OPPO 手机恢复已删除的文件?

Oppo 手机以其精美的外观和拍摄的精美照片和视频而闻名。如果您不小心丢失了 OPPO 手机中珍贵的照片、视频等重要文件&#xff0c;并且为如何找回而苦恼&#xff0c;那么您来对地方了。我们其实有很多OPPO数据恢复方案&#xff0c;现在最重要的是尽快尝试这些方法&#xff0c;防…

Git 相关内容

目录 Git 相关流程和常用命令 Git workflow Git hooks Git 相关流程和常用命令 Git远程操作详解 - 阮一峰的网络日志 Git 使用规范流程 - 阮一峰的网络日志 常用 Git 命令清单 - 阮一峰的网络日志 Git workflow 啥玩意&#xff1a; 就是一个工作流程。可以比喻成一个河流…

用逻辑回归制作评分卡

目录 一.评分卡 二.导库&#xff0c;获取数据 三.探索数据与数据预处理 1.去除重复值 2.填补缺失值 3.描述性统计处理异常值 4.为什么不统一量纲&#xff0c;也不标准化数据分布 5.样本不均衡问题 6.分训练集和测试集 三.分箱 1.分多少个箱子才合适 2.分箱要达成什么…

Antlr4: 为parser rule添加label

1. parser rule中的label 1.1 简介 Antrl4语法文件Calculator.g4&#xff0c;stat和expr两个parser rule含有多个rule element&#xff0c;我们这两个parse rule的每个rule element添加了Alternative labels&#xff08;简称label&#xff09; 按照Antlr4的语法规则&#xff…

2022年显卡性能跑分排名表

2022年显卡性能跑分排名表&#xff08;数据来源于快科技&#xff09;这个版本的电脑显卡跑分榜第一的是NVIDIA GeForce RTX 3090 Ti显卡。由于显卡跑分受不同的测试环境、不同的显卡驱动版本以及不同散热设计而有所不同&#xff0c;所以显卡跑分会一直变化。 前二十名的台式电…