双向链表<数据结构 C版>

目录

关于链表的分类

 双向链表结构体

初始化

尾插

头插

打印

判断是否为空

尾删

头删

查找

指定位置之后的插入

指定位置的删除

销毁


关于链表的分类

根据链表的三大特性,单向or双向、带头or不带头、循环or不循环,可将链表分为2*2*2,8种链表,前面我们已经实现了单链表,即:不带头单向非循环链表,它的结构简单,不常用于单独存储数据,而是作为其他数据结构的子结构。

实际运用中,只有单链表(不带头单向非循环链表)和双向链表(带头双向循环链表)运用最多,带头双向循环链表,结构最复杂,常常运用于单独存储数据,使用的链表结构也几乎都是双向带头链表。

附上一张bit课件的图:


 双向链表结构体

放一张bi课件的图片很形象:

//重定义一下类型,方便统一修改
typedef int LNDataType;typedef struct ListNode {LNDataType data;//数据struct ListNode* prev;//前一个结点struct ListNode* next;//后一个结点
}LN;

初始化

双向链表的初始化,应创建一个哨兵结点(也称头结点),它存放的数据是无效数据(假定-1),

所以我们先实现一个创建单节点的函数:

//创建新节点
LN* LNBuyNode(LNDataType x) {LN* node = (LN*)malloc(sizeof(LN));//开辟空间if (!node) {//判断为空perror("malloc fail!");exit(1);}node->data = x;//传入数据node->next = node->prev = node;//双向循环链表单节点也应满足循环,不能初始化为NULL;return node;
}

接着我们就可方便地调用,创建一个哨兵结点:

//初始化
void LNInit(LN** pphead) {assert(pphead);*pphead=LNBuynode(-1);
}

尾插

//尾插
void LNPushBack(LN* phead, LNDataType x) {assert(phead);LN* node = LNBuyNode(x);//创建新结点node->next = phead;//先对新申请的结点参数进行操作,防止对原链表造成改变node->prev = phead->prev;phead->prev->next = node;//更改尾结点的next参数的指向phead->prev = node;//更改头结点prev结点的指向
}

运行测试一下:


头插

注意头插的操作是在哨兵位后插入,双向链表为空的情况也是只剩下哨兵位,因为哨兵位并没有存储有效数据。

//头插
void LNPushFront(LN* phead, LNDataType x) {assert(phead);LN* node = LNBuyNode(x);//创建新结点node->next = phead->next;//先对新申请的结点参数进行操作,防止对原链表造成改变node->prev = phead;phead->next->prev = node;//更改头结点的下一个结点的prev指向phead->next = node;//更改头结点的next指向
}

运行测试一下:


打印

//打印
void LNPrint(LN* phead) {assert(phead);LN* pcur = phead->next;//因为头结点内存储的是无效数据,所以我们让它指向下一个结点while (pcur!=phead) {//与头结点相遇说明我们已经遍历完整个链表了printf("%d->", pcur->data);pcur = pcur->next;}printf("\n");
}

判断是否为空

双向链表为空的情况是只有一个哨兵结点而不是NULL

//判断是否为空
bool LNEmpty(LN* phead) {assert(phead);return phead->next == phead;
}

尾删

//尾删
void LNPopBack(LN* phead) {assert(phead);assert(!LNEmpty(phead));//删除操作至少有一个有效数据//LNEmptys是空返回true,取反保证不为空LN* del = phead->prev;//保存要删除的结点phead->prev = del->prev;//对要受影响的结点的参数进行更改phead->prev->next = phead;free(del);//释放掉该地址del = NULL;
}

运行测试一下:


头删

//头删
void LNPopFront(LN* phead) {assert(phead);assert(!LNEmpty(phead));LN* del = phead->next;//保存要删除的结点phead->next = del->next;//对要受影响的结点的参数进行更改phead->next->prev = phead;free(del);//释放掉该地址del = NULL;
}

运行测试一下:


查找

因为后面涉及到任意位置的操作,所以这里要写一个查找方法:

//查找
LN* LNFind(LN* phead, LNDataType x) {assert(phead);assert(!LNEmpty(phead));LN* pcur = phead->next;while (pcur!=phead) {//判断条件为!=phead,遇到哨兵位说明已经遍历完if (pcur->data == x) {return pcur;}pcur = pcur->next;}return NULL;
}

运行测试一下:


指定位置之后的插入

//指定位置之后的插入
void LNInsert(LN* pos, LNDataType x) {assert(pos);LN* node = LNBuyNode(x);node->next = pos->next;//先对要受影响的结点的参数进行更改node->prev = pos;pos->next->prev = node;//更改pos结点的后一个结点的prev参数pos->next = node;//更改pos结点的next参数
}

运行测试一下:


指定位置的删除

//任意位置的删除
void LNErase(LN* pos) {assert(pos);pos->next->prev = pos->prev;pos->prev->next = pos->next;free(pos);pos = NULL;
}

运行测试一下:


销毁

//销毁
void LNDestory(LN** phead) {assert(phead && *phead);LN* pcur = (*phead)->next;while (pcur != *phead) {LN* next = pcur->next;free(pcur);pcur = next;}free(*phead);*phead =pcur= NULL;
}

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

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

相关文章

《昇思25天学习打卡营第23天|RNN实现情感分类》

使用RNN进行情感分类:基于IMDB数据集的LSTM应用 引言 情感分析是自然语言处理(NLP)中的一个重要应用,广泛用于电影评论、社交媒体等文本数据的情感分类任务。本文将介绍如何使用递归神经网络(RNN)实现情感…

人工智能与社交变革:探索Facebook如何领导智能化社交平台

在过去十年中,人工智能(AI)技术迅猛发展,彻底改变了我们与数字世界互动的方式。Facebook作为全球最大的社交媒体平台之一,充分利用AI技术,不断推动社交平台的智能化,提升用户体验。本文将深入探…

昇思25天学习打卡营第02天|张量Tensor

这节学习的张量(Tensor)的内容进行总结 : 1、张量的概念:张量是一种多线性函数,可以表示矢量、标量和其他张量之间的线性关系。张量是MindSpore网络运算中的基本数据结构,类似于数组和矩阵。 2、可以通过直…

基于SpringBoot+Vue的校园疫情防控系统(带1w+文档)

基于SpringBootVue的校园疫情防控系统(带1w文档) 基于SpringBootVue的校园疫情防控系统(带1w文档) 主要对首页、个人中心、学生管理、疫情动态管理、知识信息管理、防疫教育管理、健康打卡管理、请假申请管理、出校登记管理、入校登记管理、核酸报告管理、交流论坛、系统管理的…

PHP 高性能框架 Workerman 凭什么能硬刚 Swoole ?

大家好,我是码农先森。 一次偶然看到了国外某机构针对 PHP 周边生态框架及扩展的性能测试排行榜,看到 Workerman 竟遥遥领先 Swoole。在我们 PHP 程序员现有的认知里,Swoole 作为一个基于 C/C 语言编写的扩展程序,性能居然落后了…

MySQL:库表操作

MySQL:库表操作 库操作查看创建字符编码集 删除修改备份 表操作创建查看删除修改 库操作 查看 查看存在哪些数据库: show databases;示例: 查看自己当前处于哪一个数据库: select database();示例: 此处由于我不处于任…

java启动shell脚本

检查demo.jar进程是否在运行 PID$(ps aux | grep ‘demo-v1’ | grep -v grep | awk ‘{print $2}’) if [ -z “$PID” ]; then echo “demo-v1 进程未运行” else echo “正在杀死 demo-v1 进程 $PID” kill -9 $PID fi rm /app/demo-v1.jar -rf mv /app/bak/demo-v1.jar …

全国区块链职业技能大赛国赛考题前端功能开发

任务3-1:区块链应用前端功能开发 1.请基于前端系统的开发模板,在登录组件login.js、组件管理文件components.js中添加对应的逻辑代码,实现对前端的角色选择功能,并测试功能完整性,示例页面如下: 具体要求如下: (1)有明确的提示,提示用户选择角色; (2)用户可看…

图像分类算法概述:深度学习方法

图像分类算法概述:深度学习方法 图像分类是计算机视觉中的一个基本任务,近年来随着深度学习的发展,图像分类算法取得了巨大的进步。本文将概述主要的深度学习图像分类算法。 #mermaid-svg-hRUH32Xj8pn0L9ug {font-family:"trebuchet ms…

关于redis单线程却能支持高并发业务的原因

Redis 虽然是单线程的,但它能够支持大并发主要是因为以下几个原因: 非阻塞 I/O:Redis 使用了非阻塞的 I/O 处理方式。在执行输入输出操作时,Redis 会向操作系统发出非阻塞式的系统调用(如 epoll 或 kqueue)…

代理 IP 的注意事项

1.代理 IP 可能存在不稳定性 由于代理 IP 是通过网络连接到远程服务器的,因此可能会存在网络不稳定的情况。在使用代理 IP 的过程中,我们需要注意监测代理 IP 是否正常工作,如果出现问题需要及时更换代理 IP。 2.代理 IP 可能存在安全问题 由…

【MySQL-20】关于触发器,你需要知道这些——>定义(创建&查看&删除) 触发器,将变更日志插入日志表中

前言 大家好吖,欢迎来到 YY 滴MySQL系列 ,热烈欢迎! 本章主要内容面向接触过C的老铁 主要内容含: 欢迎订阅 YY滴C专栏!更多干货持续更新!以下是传送门! YY的《C》专栏YY的《C11》专栏YY的《Lin…

记录一下在Hyper-v中动态磁盘在Ubuntu中不完全用到的问题(扩展根目录)

在之前给hyper虚拟机的Ubuntu分配磁盘有20G; 后来在Ubuntu中查看磁盘发现有一个分区没用到: 贴的图片是完成扩展后的 之前这里是10G,然后有个dev/sda4的分区,也是10G,Type是Microsoft Basic Data; …

植物神经紊乱?别担心,跟我一起轻松锻炼放松吧!

🌼哈喽,小伙伴们!最近我遇到了一个小问题,就是植物神经紊乱,搞得我整个人都不好了😣。但是,我可是个爱生活、爱自己的小太阳,怎么可能轻易被打败呢?🌞 &#…

私家车位共享系统小程序的设计

管理员账户功能包括:系统首页,个人中心,用户管理,车辆信息管理,车辆类型管理,车位信息管理,订单信息管理,系统管理 微信端账号功能包括:系统首页,车辆信息&a…

分享 2 个 .NET EF 6 只更新某些字段的方法

前言 EF 更新数据时,通常情况下,是更新全部字段的,但实际业务中,更新全部字段的情况其实很少,一般都是修改其中某些字段,所以为了实现这个目标,很多程序员通常会这样作: 先从数据库…

ffmpeg中的超时控制

在FFmpeg库中,很多函数没有直接的参数可以设置超时。 那么有哪些函数可以通过设置 AVFormatContext 的 interrupt_callback 来实现超时控制? avformat_open_input: 打开输入文件或流。这个函数会阻塞,尤其是在网络流的情况下&…

Robot Operating System——借用内存型消息

大纲 功能和工作原理源码分析POD特点POD 类型的优点 非POD特点 生成并发布“借用内存型消息”POD类型非POD类型 在ROS 2中,"loaned message"是一种消息传递机制,用于在发布者(publisher)和订阅者(subscriber…

HarmonyOS应用开发者高级认证,Next版本发布后最新题库 - 单选题序号2

基础认证题库请移步:HarmonyOS应用开发者基础认证题库 注:有读者反馈,题库的代码块比较多,打开文章时会卡死。所以笔者将题库拆分,单选题20个为一组,多选题10个为一组,题库目录如下,…

NFT革命:数字资产的确权、营销与元宇宙的未来

目录 1、NFT:数字社会的数据确权制度 2、基于低成本及永久产权的文化发现 3、PFP:从“小图片”到“身份表达”,再到社区筛选 4、透明表达:NFT 在数字化营销中的商业价值 5、可编程性:赋予 NFT 无限可能的应用 5.…