深入理解链表(SList)操作

目录:

  • 一、 链表介绍
    • 1.1、 为什么引入链表
    • 1.2、 链表的概念及结构
    • 1.3、 链表的分类
  • 二、 无头单向非[循环链表](https://so.csdn.net/so/search?q=循环链表&spm=1001.2101.3001.7020)的实现
    • 2.1、 [单链表](https://so.csdn.net/so/search?q=单链表&spm=1001.2101.3001.7020)的定义
    • 2.2、 动态申请一个节点
    • 2.3、 销毁(释放)所有节点
    • 2.4、 打印单链表
    • 2.5、 单链表尾插
    • 2.6、 单链表头插
    • 2.7、 单链表尾删
    • 2.8、 单链表头删
    • 2.9、 在单链表中查找指定值的节点
    • 2.10、 单链表在pos位置之后插入
    • 2.11、 单链表删除指定pos位置的节点
    • 2.12、 单链表删除指定pos位置之后的节点
    • 2.13、 求单链表长度
    • 2.14、 判断单链表是否为空
  • 三、 总结

学习链表之前,建议先学习下顺序表,下面是顺序表的直达链接:

深入理解顺序表(SeqList)

一、 链表介绍

1.1、 为什么引入链表

  • 学习链表之前,先让我们来思考一个问题:

为什么有了顺序表,还需要有链表这样的数据结构呢?

  • 顺序表存在的一些问题:
  1. 顺序表在中间/头部的插入删除,要挪动很多数据,时间复杂度为O(N),效率太低了。
  2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。
  3. 增容一般是一次增长2倍,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们
    再继续插入了5个数据,后面没有数据插入了,那么就浪费了95个数据空间SeqList。
  • 为了更好的解决上述问题,引入了链表。

1.2、 链表的概念及结构

  • 概念

前面学习的顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,而链表是一种物理存储结构上不连续的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的,可以实现更加灵活的动态内存管理。

  • 链表的组成

链表由一系列结点(链表中每一个元素称为结点)组成,结点可以在运行时动态生成。

每个结点包括两个部分:

1、数据域:存储数据元素

2、指针域:存储下一个结点地址

  • 链表的物理结构

可以看到,4个节点的地址并不是连续的,链表在物理结构上不一定是线性的,而在逻辑结构上是线性的

image-20210824110616542

  • 链表的逻辑结构(想象出来的)

image-20210822225322883

  • 注意

1、链式结构在逻辑上是连续的,但在物理上不一定连续

2、链表的节点是在堆上申请出来的

1.3、 链表的分类

链表的结构非常多样化

  • 单向、双向

image-20210827235931988

  • 带头结点、不带头节点(哨兵位的头节点,不存储有效数据)

image-20210828002001494

  • 非循环、循环

image-20210827235906305

  • 常用的两种结构

image-20210828000023080

二、 无头单向非循环链表的实现


2.1、 单链表的定义

image-20210831212953194

typedef int SLTDataType;//定义单链表节点
typedef struct SListNode
{SLTDataType data;        //数据域struct SListNode* next;  //指针域
}SLTNode;

2.2、 动态申请一个节点

//动态申请一个节点
SLTNode* BuySLTNode(SLTDataType x)
{SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));if (newnode == NULL)  //检查是否开辟成功{perror("malloc fail");return;}newnode->data = x;newnode->next = NULL;return newnode;
}

2.3、 销毁(释放)所有节点

//销毁单链表中所有节点
void SLTDestory(SLTNode** pphead)
{assert(pphead);SLTNode* cur = *pphead;while (cur != NULL)  //遍历链表{SLTNode* next = cur->next;  //保存cur的下一个节点free(cur);  //释放节点cur = next;}*pphead = NULL;  //z
}
  • 作用:

销毁单链表中所有节点,指向头节点的指针置空,防止内存泄露等问题


2.4、 打印单链表

//打印单链表
void SLTPrint(SLTNode* phead)
{SLTNode* ptr = phead;while (ptr != NULL){printf("%d->", ptr->data);ptr = ptr->next;}printf("NULL\n");
}

2.5、 单链表尾插

  • 先来看一种错误写法:这个也是弄懂单链表的关键

传一级指针的值,用一级指针接收

  • 这种写法会导致一个问题:

因为当链表为空时,我们需要改变 plist 的指向,使其指向第一个节点。

而初始 plist 和 phead 都指向 NULL,调用函数后,phead 指向了新的节点,而 plist 还是指向 NULL 的。

  • 如何解决呢:

plist 是指向第一个节点的指针,想要在函数中改变 plist 的值(指向),必须要把 plist指针的地址 作为实参传过去,形参用 二级指针 接收,这样在函数中对二级指针解引用得到 plist 的值,就可以改变 plist 的值(指向)了

记住:在函数里面要改变 int,则要传 int* ,要改变 int* ,则要传 int**

  • 注意区分开这几个代表的意思哦

平时一般不用解引用两层,没啥意义,一般是解引用一层,来改变外面 plist 的指向。

  • **正确写法:**通过二级指针改变外面 plist 的指向

单链表为空时,plist 直接指向新节点;

单链表不为空时,先找到单链表的尾节点,然后将尾节点的next指向新节点

//单链表尾插
void SLTPushBack(SLTNode** pphead, SLTDataType x)
{assert(pphead);  //检查参数是否传错SLTNode* newnode = BuySListNode(x);  //动态申请一个节点if (*pphead == NULL)  //单链表中没有节点时{*pphead = newnode;  //plist指向新节点}else if (*pphead != NULL)  //单链表中已经有节点时{SLTNode* tail = *pphead;while (tail->next != NULL)  //找到单链表中的最后一个节点{tail = tail->next;}tail->next = newnode;  //最后一个节点的next指向新节点}
}
  • 运行结果如下:

在这里插入图片描述


2.6、 单链表头插

  • 图解头插操作

image-20210824230705865

  • 代码如下

因为要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址

//单链表头插
void SLTPushFront(SLTNode** pphead, SLTDataType x)
{assert(pphead);  //检查参数是否传错SLTNode* newnode = BuySListNode(x);  //动态申请一个节点newnode->next = *pphead;  //新节点的next指针指向plist指向的位置*pphead = newnode;  //plist指向头插的新节点
}
  • 运行结果如下:

在这里插入图片描述


2.7、 单链表尾删

  • 图解尾删操作

image-20210824224625868

  • 代码如下

单链表只有一个节点时,删除节点,plist 指向 NULL;

单链表有多个节点时,先找到单链表尾节点的上一个节点,删除尾节点,然后将该节点的next指向 NULL;

因为可能要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址。

//单链表尾删
void SLTPopBack(SLTNode** pphead)
{assert(pphead);   //检查参数是否传错assert(*pphead);  //断言,链表不能为空SLTNode* tail = *pphead;if ((*pphead)->next == NULL)  //链表只有一个节点{free(*pphead);   //删除节点*pphead = NULL;  //plist置空}else  //链表中有多个节点{while (tail->next->next != NULL)  //找到链表的尾节点的上一个节点{tail = tail->next;}free(tail->next);   //删除尾节点tail->next = NULL;  //置空/*思路2:SLTNode* prev = *pphead;while (tail->next)  //找到链表的尾节点和它的上一个节点{prev = tail;tail = tail->next;}free(tail);         //删除尾节点prev->next = NULL;  //置空*/}
}
  • 运行结果如下:

在这里插入图片描述


2.8、 单链表头删

  • 图解头删操作

image-20210825001225011

  • 代码如下

因为可能要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址。

//单链表头删
void SLTPopFront(SLTNode** pphead)
{assert(pphead);   //检查参数是否传错assert(*pphead);  //链表不能为空SLTNode* cur = *pphead;  //保存头节点的地址*pphead = cur->next;  //plist指向头节点的下一个节点free(cur);  //删除头节点
}
  • 运行结果如下:

在这里插入图片描述


2.9、 在单链表中查找指定值的节点

如果查找到,返回该节点的地址;没有查找到,返回 NULL。

//在单链表中查找指定值节点
SListNode* SLTFind(SLTNode* phead, SLTDataType x)
{SLTNode* cur = phead;//遍历链表while (cur != NULL){if (cur->data == x){return cur;  //找到了,返回该节点的地址}cur = cur->next;}return NULL;  //未找到,返回NULL
}

2.10、 单链表在pos位置之后插入

  • 图解插入操作

image-20210827234957091

  • 分析思考为什么不在pos位置之前插入?

单链表不适合在pos位置之前插入,因为需要遍历链表找到pos位置的前一个节点,

单链表更适合在pos位置之后插入,如果在后面插入,只需要知道pos位置就行,会简单很多

C++官方库里面单链表给的也是在之后插入

  • 代码如下
//单链表在指定pos位置之后插入
void SLTInsertAfter(SLTNode* pos, SLTDataType x)
{assert(pos);  //给的pos位置不能为空SLTNode* newnode = BuySListNode(x);  //动态申请一个节点newnode->next = pos->next; //新节点的next指针指向pos位置后一个节点pos->next = newnode;       //pos位置的next指向新节点
}
  • 作用:

先调用 SLTFind 函数在单链表中查找指定值的节点,查找到了,返回该节点的地址,也即是我们指定的 pos 位置,然后将其传给 SLTInsertAfter 函数。


2.11、 单链表删除指定pos位置的节点

  • 图解删除操作

image-20210827235619437

  • 代码如下

要考虑到两种情况:pos位置为单链表的第一个节点,pos位置为中间节点;

因为可能要改变外部 plist 的指向,所以用二级指针接收指针 plist 的地址。

//单链表删除指定pos位置的节点
void SLTErase(SLTNode** pphead, SLTNode* pos)
{assert(pphead);assert(*pphead); //链表不能为空assert(pos);     //给的pos位置不能为空//pos位置为第一个节点,相当于头删if (pos == *pphead){SLTPopFront(pphead);}//pos位置为中间节点else{SLTNode* prev = *pphead;while (prev->next != pos)  //找到pos位置的前一个节点{prev = prev->next;}prev->next = pos->next;  //pos位置的前一个节点指向pos位置的后一个节点free(pos);  //释放pos节点pos = NULL; //置空}
}
  • 补充一下,assert 断言

assert 放在函数里面检查参数,一方面是为了安全,另一方面也是为了防止有人调用该函数时,不正确的使用,错误传入参数,好及时提醒到他,写代码时一定要考虑到有人不正确的使用该函数时的场景,来进行避免

  • 测试其功能

先调用 SLTFind 函数在单链表中查找指定值的节点,查找到了,返回该节点的地址,也即是我们指定的 pos 位置,然后将其传给 SLTErase 函数。


2.12、 单链表删除指定pos位置之后的节点

  • 图解删除操作

在这里插入图片描述

  • 代码如下
//单链表删除指定pos位置之后的节点
void SLTEraseAfter(SLTNode* pos)
{assert(pos);       //给的pos位置不能为空assert(pos->next); //给的pos位置不能是尾节点SLTNode*del = pos->next;  //保存pos位置的后一个节点pos->next = pos->next->next;free(del);  //释放pos位置的后一个节点del=NULL;
}

-运行结果如下:

在这里插入图片描述


2.13、 求单链表长度

//求单链表长度
int SLTSize(SLTNode* phead)
{int size = 0;SLTNode* cur = phead;while (cur != NULL)  //遍历链表{size++;cur = cur->next;}return size;
}

2.14、 判断单链表是否为空

plist为空,返回 1 (true),非空,返回 0 (false)

//单链表判空
bool SLTEmpty(SLTNode* phead)
{//plist为空,返回1(true),非空,返回0(false)return phead == NULL;/*写法2:return phead == NULL ? true : false;*/
}

三、 总结

单链表是一种简单且高效的数据结构,适用于需要频繁插入和删除操作的场景。通过掌握单链表的创建、插入、删除、查找、修改和遍历等操作,我们可以更好地理解和应用这一数据结构,从而提高程序的效率和可靠性。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

论文阅读:OpenSTL: A Comprehensive Benchmark of Spatio-Temporal Predictive Learning

论文地址:arxiv 摘要 由于时空预测没有标准化的比较,所以为了解决这个问题,作者提出了 OpenSTL,这是一个全面的时空预测学习基准。它将流行的方法分为基于循环和非循环模型两类。OpenSTL提供了一个模块化且可扩展的框架&#xff…

澳鹏干货 | 大语言模型的上下文窗口 (Context Windows)

大语言模型(LLMs)极大地提升了人工智能在理解和生成文本方面的能力。其中一个影响其效用的重要方面是“上下文窗口”(Context Windows)—— 这个概念直接影响着模型接收和生成语言的有效性。 本期澳鹏干货将深入探讨上下文窗口对…

如何让员工参与到精益变革的持续改进中?

实践证明,精益变革并非一蹴而就,它需要全员参与、持续改进,才能真正将精益理念融入企业的血脉之中。那么,如何让员工积极参与到精益变革的持续改进过程中呢?深圳天行健TPM管理咨询公司解析如下: 一、构建精…

电力电子技术(一)

变压器漏感对整流电路的影响:

CSS弹性布局

Flex 是 Flexible Box 的缩写,意为“弹性布局”或者“弹性盒子”,是 CSS3 中的一种新的布局模式,可以简便、完整、响应式地实现各种页面布局,当页面需要适应不同的屏幕大小以及设备类型时非常适用。目前,几乎所有的浏览…

IntelliJ IDEA中配置scala

1.IDEA中 配置 maven 左上角 file -> Setting 选择(或直接搜maven) Build, Execution,Deployment -> Build Toos -> Maven Maven home path 选择 maven 安装目录(bin的上层目录) 示例: D:\maven\apache-maven-3.8.6 User settings…

python异常检测 - 随机离群选择Stochastic Outlier Selection (SOS)

python异常检测 - Stochastic Outlier Selection (SOS) 前言 随机离群选择SOS算法全称stochastic outlier selection algorithm. 该算法的作者是jeroen janssens. SOS算法是一种无监督的异常检测算法. 随机离群选择SOS算法原理 随机离群选择SOS算法的输入: 特征矩阵(featu…

git的学习使用(搭建本地仓库,创建本地仓库,配置本地仓库)(附带Ubuntu云服务器git安装流程)

学习目标: 学习使用git,并且熟悉git的使用 学习内容: 必备环境:xshell,Ubuntu云服务器 如下: 搭建 git 环境 搭建git环境: 1、先检查自己的云服务上是否有已经安装好的git,这里…

AGI|如何构建一个RAG应用?入门新手攻略!

目录 一、概述 二、过程概述 三、如何优化提问? 四、路由和高级查询 五、丰富索引结构 六、重排序上下文 七、总结 一、概述 Retrieval Augmented Generation RAG 检索增强的内容生成。 从字面上来看检索只是一种手段途径,在人工智能领域中存在多种…

什么是CGI?

什么是CGI? ‌CGI,全称为“通用网关接口”(Common Gateway Interface),是一种用于Web服务器与应用程序之间通信的标准接口。它可以让Web服务器调用应用程序来执行特定任务,并将结果返回给Web浏览器。 CGI描…

车载电源OBC+DC/DC

文章目录 1. 车载DC/DC应用场景2. PFC2.1 简介2.2 专业名词2.3 常见拓扑结构2.3.1 传统桥式PFC2.3.2 普通无桥型PFC2.3.3 双Boost无桥PFC2.3.4 图腾柱PFC2.3.5 参考资料 2.4 功率因数2.4.1 简介2.4.2 计算 3. DC/DC3.1 Boost升压电路3.1.1 简介3.1.2 电路框图3.1.3 工作原理3.1…

解锁编程的力量:SPL的学习之旅

SPL 一、前言二、集算器应用场景三、下载四、集算器的基本使用 一、前言 一种面向结构化数据的程序计算语言 集算器又称:SPL(Structured Process Language) 敏捷计算是集算器的主要特征 二、集算器应用场景 数据准备(跑批&…

通过观测云 DataKit Extension 接入 AWS Lambda 最佳实践

前言 AWS Lambda 是一项计算服务,使用时无需预配置或管理服务器即可运行代码。AWS Lambda 只在需要时执行代码并自动缩放。借助 AWS Lambda,几乎可以为任何类型的应用程序或后端服务运行代码,而且无需执行任何管理。 Lambda Layer 是一个包…

yakit使用教程(四,信息收集)

本文仅作为学习参考使用,本文作者对任何使用本文进行渗透攻击破坏不负任何责任。 前言:yakit下载安装教程。 一,基础爬虫。 在新建项目或新建临时项目后,点击安全工具,点击基础爬虫。 此工具并不是为了爬取网站上的一…

Protobuf:消息更新

Protobuf:消息更新 更新字段保留字段未知字段option选项 在开发中,需要对产品进行版本迭代。迭代前后,类的成员可能就会有所改动,一旦类成员改动,那么老版本的对象,新版本可能就无法解析,此时就…

一文了解 Linux 系统的文件权限管理

文章目录 引入Linux文件权限模型查看文件权限权限信息解析修改文件权限符号模式八进制数字模式 引入 在Linux操作系统中,我们想查看我们对文件拥有哪些权限时,可以在终端键入ls -l或ll命令,终端会输出当前路径下的文件信息,如文件…

【网络】【Linux】多路转接技术

多路转接技术 文章目录 1.select1.1select系统调用及参数介绍1.2select基本工作流程1.3select技术实现echo服务器1.4select优缺点1.5select的适用场景 2.poll(了解)2.1poll系统调用及参数介绍2.2poll技术实现echo服务器2.3poll优缺点 3.epoll3.1epoll系…

【新人系列】Python 入门(二):Python IDE 介绍

✍ 个人博客:https://blog.csdn.net/Newin2020?typeblog 📝 专栏地址:https://blog.csdn.net/newin2020/category_12801353.html 📣 专栏定位:为 0 基础刚入门 Python 的小伙伴提供详细的讲解,也欢迎大佬们…

【Windows命令】Windows下启动Nginx后,在任务管理器里面没有发现nginx.exe进程

如题,当在本地Windows环境下想用反向代理时,突然发现在任务管理器里面没有发现nginx.exe进程,但是端口又是占用的。这时就要用Windows命令了。 查询端口占用 netstat -ano | findstr :80 根据进程ID(pid)查询进程名称…

ESP32移植Zephyr RTOS(一)-----hello world

硬件平台:实战派ESP32-C3开发板 zephyr版本:Zephyr version 3.7.99 开发环境:ubuntu 24.4 之前一直想用正点原子阿波罗F4来写zephyr系列教程来自,但是本人水平有限RGB LCD实在是搞不懂,遂放弃,正好手头有一…