单链表原来是这样实现的!

文章目录

  • 前言
  • 1. 链表的概念及结构
    • 1.1在链表里,每节“车厢”是什么样的呢?
    • 1.2为什么还需要指针变量来保存下⼀个节点的位置?
  • 2. 单链表的实现
    • 1. 定义结构体`(Seqlist)`
    • 2. 打印函数`(SLTPrint)`
    • 小插曲,创建节点函数`CreateNode`
    • 3. 尾插函数 `(SLTPushBack)`
    • 4. 头插函数 `(SLTPushFront)`
    • 5. 尾删函数(`SLTPopBack`)
    • 6. 头删函数(`SLTPopFront`)
    • 小插曲,pos查找函数` SLTFind`
    • 7. “插入指定位置前”函数(`SLTInster`)
    • 8.“删除指定位置后”函数
    • 9.销毁单链表函数`SLTDestroy`
  • 结语

前言

“我会定期分享我的学习经验,也欢迎大家留言和交流,让我们共同学习和进步!感谢大家的支持,让我们一起开启这段充满技术乐趣的旅程吧!”


1. 链表的概念及结构

概念:链表是⼀种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的

在这里插入图片描述

链表的结构跟火车车厢厢相似,淡季时车次的车厢会相应减少,旺季时车次的车厢会额外增加几节。只需要将火车里的某节节厢去掉/加上,不会影响其他车厢,每节车厢都是独立存在的。
车厢是独里存在的,且每节车厢都有车门。想象⼀下这样的场景,假设每节车厢的车门都是锁上的状态,需要不同的钥匙才能解锁,每次只能携带⼀把钥匙的情况下如何从车头走到车尾?
最简单的做法:每节车厢里都放⼀把下一节车厢的钥匙。

1.1在链表里,每节“车厢”是什么样的呢?

在这里插入图片描述

与顺序表不同的是,链表⾥的每节"车厢"都是独立申请下来的空间,我们称之为“结点/节点”,节点的组成主要有两个部分:当前节点要保存的数据和保存下一个节点的地址(指针变量)。
图中指针变量 plist保存的是第⼀个节点的地址,我们称plist此时“指向”第⼀个节点,如果我们希望plist“指向”第⼆个节点时,只需要修改plist保存的内容为0x0012FFA0。

1.2为什么还需要指针变量来保存下⼀个节点的位置?

链表中每个节点都是独立申请的(即需要插⼊数据时才去申请⼀块节点的空间),我们需要通过指针
变量来保存下⼀个节点位置才能从当前节点找到下⼀个节点。


2. 单链表的实现

1. 定义结构体(Seqlist)

在SList.h头文件中

typedef int SLNDataType;
typedef struct SListNode
{SLNDataType val;struct SList* next;//这里只是指针,不是结构体
}SLNode;

2. 打印函数(SLTPrint)

注意下述代码皆是:
SList.h头文件中定义函数
SList.c文件中实现函数
Test.c文件中函数测试

SeqList.h文件中
定义函数:
在这里插入图片描述

SList.c文件中
实现函数:

void SLTPrint(SLNode* phead) //打印单链表
{SLNode* cur = phead;while (cur != NULL){printf("%d->", cur->val);cur=cur->next;}printf("NULL");
}

小插曲,创建节点函数CreateNode

在实现下面的插入函数之前,还需要一个函数来开辟空间给新的节。
函数实现

SLNode* CreateNode(SLNDataType x) //新建节点,开辟空间
{SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));if (newnode == NULL){perror("malloc fail");exit(-1);}newnode->val = x;newnode->next = NULL;return newnode;
}

3. 尾插函数 (SLTPushBack)

定义函数:

在这里插入图片描述

实现函数:

void SLTPushBack(SLNode** pphead, SLNDataType x) //尾插
{SLNode* newnode = CreateNode(x);if (* pphead == NULL){*pphead = newnode;}else{SLNode* tail = * pphead; //找尾while (tail->next != NULL){tail = tail->next;  //因为tail是局部变量,而tail->next是结构体,出来作用域tail就销毁了}tail->next = newnode; //所以这里把newnode赋值给tail->next}
}

函数测试:

int main()
{SLNode* plist = NULL;SLTPushBack(&plist, 1);SLTPushBack(&plist, 2);SLTPushBack(&plist, 3);SLTPrint(plist);return 0;
}

运行结果:
![在这里插入图片描述](https://img-blog.csdnimg.cn/4b96a69eca1243a1aa4a0e9ebf888f87.png


4. 头插函数 (SLTPushFront)

定义函数:

![在这里插入图片描述](https://img-blog.csdnimg.cn/6c514f963f144d65a681e2c56dad01ac.png

实现函数:

void SLTPushFront(SLNode** pphead, SLNDataType x) //头插
{ SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));newnode->next =* pphead;newnode->val = x;*pphead = newnode;
}

函数测试:

int main()
{SLNode* plist = NULL;SLTPushFront(&plist,520 );SLTPushBack(&plist,1);SLTPushBack(&plist,1);SLTPushFront(&plist,520);SLTPrint(plist);return 0;
}

运行结果:
在这里插入图片描述


5. 尾删函数(SLTPopBack)

定义函数:

在这里插入图片描述

实现函数:

void SLTPopBack(SLNode** pphead)  //尾删
{assert(pphead);assert(*pphead);if ((*pphead)->next== NULL){free(*pphead);*pphead = NULL;}else{SLNode* prev = NULL;SLNode* tail = *pphead;while (tail->next != NULL){prev = tail;tail = tail->next;}free(tail);tail = NULL;prev->next = NULL;}
}

函数测试:

int main()
{SLNode* plist = NULL;SLTPushFront(&plist,520 );SLTPushBack(&plist,1314);SLTPushBack(&plist,00544);SLTPopBack(&plist);SLTPrint(plist);return 0;
}

运行结果:
在这里插入图片描述


6. 头删函数(SLTPopFront)

定义函数:

![在这里插入图片描述](https://img-blog.csdnimg.cn/dff2c7abe0f7483d83f82307db1ceb3f.png

实现函数:

void SLTPopFront(SLNode** pphead) //头删
{assert(*pphead);SLNode* tail = *pphead;tail = tail->next;free(*pphead);*pphead = tail;
}

函数测试:

int main()
{SLNode* plist = NULL;SLTPushFront(&plist,5201314);SLTPushBack(&plist,00544);SLTPushBack(&plist,44944);SLTPopFront(&plist);SLTPrint(plist);return 0;
}

运行结果:
在这里插入图片描述


小插曲,pos查找函数 SLTFind

用来确定pos位置,方便后面调用
实现函数:

SLNode* SLTFind(SLNode** pphead, SLNDataType x) //pos的查找函数
{assert(pphead);SLNode* cur = *pphead;while (cur){if (cur->val == x){return cur;}cur = cur->next;}return NULL;
}

7. “插入指定位置前”函数(SLTInster)

定义函数:

![在这里插入图片描述](https://img-blog.csdnimg.cn/94333646a22546b4a4baf71c46711172.png

实现函数:

void* SLTInster(SLNode** pphead, SLNode* pos, SLNDataType x) //指定位置前面插入
{assert(pos);assert(pphead);assert(*pphead);SLNode* node = CreateNode(x);if (*pphead == pos){node->next = *pphead;*pphead = node;}SLNode* cur = *pphead;while (cur->next != pos){cur = cur->next;}cur->next = node;node->next = pos;
}

函数测试:

int main()
{SLNode* plist =NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLNode* Find = SLTFind(&plist, 3);SLTInster(&plist,Find,123);SLTPrint(plist);return 0;
}

运行结果:
如同在第一个值为3的节点前面插入了123;
在这里插入图片描述


8.“删除指定位置后”函数

定义函数:

在这里插入图片描述

实现函数:

void* SLTEraseAfter(SLNode* pos) //指定位置后面删除
{assert(pos && pos->next);SLNode* del = pos->next;pos->next = del->next;free(del);
}

函数测试:

int main()
{SLNode* plist =NULL;SLTPushBack(&plist,520);SLTPushBack(&plist,2);SLTPushBack(&plist,520);SLNode* Find = SLTFind(&plist, 2);SLTEraseAfter(Find);SLTPrint(plist);return 0;
}

运行结果:
如图在第一个值为520的节点后面删除了小3;
在这里插入图片描述


9.销毁单链表函数SLTDestroy

定义函数:
在这里插入图片描述
实现函数:

void SLTDestroy(SLNode** pphead) //销毁单链表
{assert(pphead);SLNode* cur= *pphead;while (cur){SLNode* next = cur;free(cur);cur = next;}*pphead = NULL;
}

测试函数:

int main()
{SLNode* plist =NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTDestroy(&plist);return 0;
}

结语

感谢您阅读我的博客,我希望您能从中获得一些启发和帮助。如果您喜欢这篇博客,请分享给您的朋友,也欢迎留下您的评论和反馈。您的支持是我继续分享和创作的动力。谢谢!希望我们能在未来的博客中再次相见。祝您一切顺利,期待与您再次相会!

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

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

相关文章

嵌入式的学习需要合理规划时间

低级的欲望放纵即可获得,高级的欲望只有克制才能达成。——卡耐基1、粉丝的误会 很多粉丝,问我, "胡老师我想报您的培训班。" ... 得知我知识业余时间写文章,紧接着又会问, "jg单位这么清闲啊&#…

预制构件二维码如何生成?

PC预制构件张贴二维码标识牌,可实现预制构建基本信息、设计图纸、安装说明书信息展示微信扫一扫即可查看预制件信息,大大提高施工的精度和效率;同时也可以实现预制生成过程管理、运输过程管理、安装过程管理、后期运维管理实现预制件的过程质…

Windows安装mysql8.0

官网地址:MySQL :: MySQL Community Downloads 选择相应版本信息下载 默认选择点击下一步 默认配置点击next 设置密码 默认配置

Linux系统管理:WinSCP 安装与使用

目录 一、实验 1.下载WinSCP 2.安装WinSCP 3.使用WinSCP 一、实验 1.下载WinSCP (1)地址 Downloading WinSCP-6.1.2-Setup.exe :: WinSCP 2.安装WinSCP (1)选择安装程序模式 (2)点击 (3…

机器学习库:pandas

☁️主页 Nowl 🔥专栏《机器学习实战》 《机器学习》 📑君子坐而论道,少年起而行之 文章目录 写在开头 基本数据格式 DataFrame 数据选取 iloc 数据描述 head describe 数据合并 merge 数据删除 drop drop删除多列 处理缺失…

SpringBoot进阶——解释springboot的自动配置原理

相关的博客文章如下: SpringBootApplication注解的理解——如何排除自动装配 & 分布式情况下如何自动加载 & nacos是怎么被发现的 引出 1.spring.factories文件存储能够进行自动配置的Bean信息; 2.EnableAutoConfiguration关闭数据源的自动配置…

Flask学习二:项目拆分、请求与响应、cookie

教程 教程地址: 千锋教育Flask2框架从入门到精通,Python全栈开发必备教程 老师讲的很好,可以看一下。 项目拆分 项目结构 在项目根目录下,创建一个App目录,这是项目下的一个应用,应该类似于后端的微服…

灯塔的安装

Docker 安装 docker 安装参考:https://docs.docker.com/engine/install/ shell脚本: curl -fsSL https://get.docker.com -o get-docker.sh sudo sh get-docker.sh灯塔安装 mkdir docker-ARL;cd docker-ARL curl https://bootstrap.pypa.io/get-pip.py -o get-pip…

【Linux学习】基础IO

目录 八.系统文件IO 8.1 前言 8.2 C语言文件IO C语言常用的基本函数 C语言默认打开的的三个流 8.3 系统文件IO open接口 close接口 write接口 read接口 8.4 C语言文件IO与系统文件IO的关系 八.系统文件IO 8.1 前言 系统文件 I/O(输入/输出)是指在…

pandas 如何获取dataframe的行的数量

pandas的dataframe提供了多种方法获取其中数据的行的数量,本偏文章就是介绍几种获取dataframe行和列出量的方法。 为了能够详细说明如何通过代码获取dataframe的行数和列数,需要先创建一个dataframe如下: import pandas as pdtechnologies …

C++设计模式之策略模式

策略模式 介绍示例示例测试运行结果应用场景优点总结 介绍 策略模式是一种行为设计模式。在策略模式中,可以创建一些独立的类来封装不同的算法,每一个类封装一个具体的算法,每一个封装算法的类叫做策略(Strategy),为了保证这些策…

NX二次开发UF_CURVE_ask_wrap_curve_parents 函数介绍

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan UF_CURVE_ask_wrap_curve_parents Defined in: uf_curve.h int UF_CURVE_ask_wrap_curve_parents(tag_t curve_tag, tag_t * defining_face, tag_t * defining_plane, tag_t * defin…

18. Python 数据处理之 Numpy

目录 1. 简介2. 安装和导入Numpy3. ndarray 对象4. 基本运算5. 索引、切片和迭代6. 条件和布尔数组7. 变换形状8. 操作数组 1. 简介 数据分析的流程概括起来主要是:读写、处理计算、分析建模和可视化4个部分。 Numpy 是Python 进行科学计算,数据分析时…

MySQL进阶_10.锁

文章目录 一、概述二、MySQL并发事务访问相同记录2.1、读-读2.2、写-写2.3、读-写2.4、并发问题的解决方案 三、锁的不同角度分类3.1、 读锁、写锁3.1.1、 锁定读 3.2、表级锁、页级锁、行锁3.2.1、表锁3.2.2、意向锁3.2.2.1、意向锁的作用3.2.2.2、意向锁的互斥性 3.2.3、自增…

2019年全国硕士研究生入学统一考试管理类专业学位联考英语(二)试题

文章目录 2019年考研英语二真题SectionⅠ Use of EnglishSection II Reading ComprehensionText 121——细节信息题22——细节信息题23——细节信息题24——细节信息题25——词义题 Text 226——细节信息题27——细节信息题28——细节信息题29——细节信息题30——态度题 Text …

springboot2.0 集成swagger3+Knife4j导出离线API 配置

springboot 版本2.3.1 一、集成swagger3 引入swagger依赖包 <!--swagger3集成--><dependency><groupId>org.springframework.plugin</groupId><artifactId>spring-plugin-core</artifactId><version>2.0.0.RELEASE</version>…

Linux面试题(三)

目录 34、du 和 df 的定义&#xff0c;以及区别&#xff1f; 35、awk 详解。 36、当你需要给命令绑定一个宏或者按键的时候&#xff0c;应该怎么做呢&#xff1f; 37、如果一个 linux 新手想要知道当前系统支持的所有命令的列表&#xff0c;他需要怎么做&#xff1f; 38、…

第一百八十一回 如何绘制阴影效果

文章目录 1. 概念介绍2. 使用方法2.1 SegmentedButton2.2 ButtonSegment 3. 代码与效果3.1 示例代码3.2 运行效果 4. 内容总结 1. 概念介绍 我们在本章回中介绍的SegmentedButton组件是一种分段式按钮&#xff0c;它把多个按钮连接成一组显示&#xff0c;组内再对不同的按钮进…

[计算机网络]应用层概述

0.写在前面: 该层为教学模型的最后一层,某种意义上来说是最接近各位开发者的一层,正因如此,这层中的很多定义和概念大家都有属于自己的理解, 完全按照书本反而才是异类,因此在这里我会去结合我做前端开发的一些经验,来处理和讲解一些概念,另外本层中的部分协议也不会过多阐述了…

VUE简易购物车程序

目录 效果预览图 完整代码 效果预览图 完整代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>…