【探索数据结构】线性表之双链表

🎉🎉🎉欢迎莅临我的博客空间,我是池央,一个对C++和数据结构怀有无限热忱的探索者。🙌

🌸🌸🌸这里是我分享C/C++编程、数据结构应用的乐园✨

🎈🎈🎈期待与你一同在编程的海洋中遨游,探索未知的技术奥秘💞

📝专栏指路:

📘【C++】专栏:深入解析C++的奥秘,分享编程技巧与实践。

📘【数据结构】专栏:探索数据结构的魅力,助你提升编程能力。

前言

之前我们已经探索了顺序表和单链表我们继续一起来探索逻辑结构里面的线性结构。线性表在逻辑结构上是连续的,线性表中双链表(本篇主角)在物理结构上是不连续的。

文章重点介绍:带头双向循环链表

一、双链表

1.概念

双链表,也叫双向链表,是链表的一种特殊形式。在双链表中,每个数据节点都有两个指针,一个指向前一个节点(前驱节点),另一个指向后一个节点(后继节点)。这种结构使得从双链表中的任意一个节点开始,都可以很方便地访问它的前驱节点和后继节点。

2.分类

(1)按不同属性分

  • 带头节点的双链表:这种双链表在第一个数据节点之前有一个头结点。头结点是为了操作的统一和方便而设立的,放在第一元素的结点之前,其数据域一般无意义(也可存放链表的长度)。有了头结点,对在第一元素结点前插入结点和删除第一结点,其操作与其它结点的操作就统一了。
  • 不带头节点的双链表:与带头结点的双链表相反,这种链表没有头结点,直接从第一个数据节点开始。

(2)按循环性分

  • 双向循环链表:在双向链表的基础上,将头结点的后驱指针指向尾节点,尾节点的前驱指针指向头结点,从而形成一个双向环
  • 双向非循环链表:这是标准的双向链表,没有形成一个环,只是简单地通过前驱和后继指针连接各个节点。

3.链表结构

typedef int LTDataType;
typedef struct LTNode LTNode;
struct LTNode
{LTDataType data;//数据LTNode* prev;//前驱指针LTNode* next;//后继指针
};

二、对双链表的操作

0.创建节点

//创建节点
LTNode* LTBuyNode(LTDataType x)
{LTNode* node = (LTNode*)malloc(sizeof(LTNode));if (node == NULL){perror("malloc fail!");exit(1);}node->data = x;node->next = node->prev = node;//不能置为空,都要指向自己return node;
}

1.初始化

后续对链表的操作都是不需要改变头结点的,哨兵位节点不能被删除,节点的地址,也不能发生改变只需传一级指针。为了保持接口的一致性。我们没有在初始化方法中选择传二级指针的方式实现

// 初始化
LTNode* LTInit()
{LTNode* phead = LTBuyNode(-1);return phead;
}

2.打印

// 打印
void LTPrint(LTNode* phead)
{//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->data);pcur = pcur->next;}printf("\n");
}

3.尾插

只有头结点时的尾插:

// 尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);//phead phead->prev phead->nextLTNode* newnode = LTBuyNode(x);//先改变新节点的指针指向不影响原链表newnode->prev = phead->prev;newnode->next = phead;//不可以调换下面两句顺序,否则会找不到原链表的尾结点phead->prev->next = newnode;phead->prev = newnode;
}

4.头插

// 头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);//phead phead->next LTNode* newnode = LTBuyNode(x);newnode->next = phead->next;newnode->prev = phead;//尽量不要调换下面两句顺序phead->next->prev = newnode;phead->next = newnode;
}

5.尾删

// 尾删
void LTDelBack(LTNode* phead)
{assert(phead&&phead->next);//链表不能为空//phead  phead->prev//把要删除的节点先存起来以防找不到他的前一个节点LTNode* del = phead->prev;del->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}

6.头删

// 头删
void LTDelFront(LTNode* phead)
{assert(phead && phead->next);//链表不能为空//把要删除的节点先存起来以防找不到他的后一个节点LTNode* del = phead->next;del->next->prev = phead;phead->next = del->next;free(del);del = NULL;
}

7.查找

LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){if (pcur->data == x){return pcur;}pcur = pcur->next;}return NULL;
}

8.在指定位置之后插入数据

// 指定位置之后插入
void LTPushPos(LTNode* pos, LTDataType x)
{assert(pos);//pos->next pos LTNode* newnode = LTBuyNode(x);//先改变新节点的指针指向不影响原链表newnode->prev = pos;newnode->next = pos->next;pos->next->prev = newnode;pos->next = newnode;
}

9.删除指定节点

pos理论上来说不能为phead,但是没有参数phead,无法增加校验

// 删除指定节点
void LTErase(LTNode* pos)
{assert(pos);//pos->next pos->prevpos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}

10.销毁链表

// 销毁
void LTDestroy(LTNode* phead)
{assert(phead);//从第一个有效节点遍历LTNode* pcur = phead->next;while (pcur != phead){//先把要删除节点的下一个节点存起来//不然要删除后续节点无法被找到LTNode* next = pcur->next;free(pcur);pcur = next;}//把哨兵位置为空free(phead);phead = NULL;
}

三、测试(仅供参考)

#include"List.h"
void test01()
{LTNode* plist = LTInit();//尾插LTPushBack(plist, 3);LTPushBack(plist, 2);LTPushBack(plist, 1);LTPrint(plist);//头插LTPushFront(plist, 4);LTPushFront(plist, 5);LTPushFront(plist, 6);LTPrint(plist);//尾删LTDelBack(plist);LTPrint(plist);//头删LTDelFront(plist);LTPrint(plist);LTNode* find = LTFind(plist, 5);if (find == NULL){printf("没有找到\n");}else{printf("找到了\n");}/*LTErase(find);LTPrint(plist);*/LTPushPos(find, 10);LTPrint(plist);LTDestroy(plist);//为保持接口一致性没有传二级指针//需要手动把实参置为空plist = NULL;
}
int main()
{test01();return 0;
}

下回预告:栈

持续更新中...

敬请期待

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

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

相关文章

【超全干货】一文讲清什么是全民分销?怎么做好全民分销?

一、什么是全民分销? 全民分销,作为新时代营销模式的代表之一,是基于互联网尤其是社交媒体平台兴起的一种分销策略。它打破了传统零售与电子商务的界限,允许任何个人,无论是否为专业销售人员,都能成为品牌…

【WEB前端2024】开源智体世界:乔布斯3D纪念馆-第27课-门的打开

【WEB前端2024】开源智体世界:乔布斯3D纪念馆-第27课-门的打开 使用dtns.network德塔世界(开源的智体世界引擎),策划和设计《乔布斯超大型的开源3D纪念馆》的系列教程。dtns.network是一款主要由JavaScript编写的智体世界引擎&am…

FreeRTOS_互斥量_学习笔记

互斥量 数值只有0或1 谁获得互斥量,就必须由谁释放同一个互斥量。 但其实在freeRTOS中,任务A获取的互斥锁,任务B也能释放。因此谁上锁谁开锁只是约定,在程序实现上不是强制的。 “可重入的函数"是指:多个任务同时…

Qt输入输出类使用总结

Qt输入输出类简介 QTextStream 类(文本流)和 QDataStream 类(数据流)Qt 输入输出的两个核心类,其作用分别如下: QTextStream 类:用于对数据进行文本格式的读/写操作,可在 QString、QIODevice或 QByteArray 上运行,比如把数据输出到 QString、QIODevice 或 QByteArray 对象…

Linux-文件或目录权限

在使用 ll 时,可以查看文件夹内容的详细信息,信息的第1位表示类型,具体信息如下: 类型说明-普通文件d文件夹b块设备文件c字符设备文件p管道文件s套接口文件 第2-10位表示权限, 举例:rwxr-xr-x 类型说明r…

业务架构核心要素之间的关系

背景 前面已经对业务架构的概念和发展简史有了初步的了解,现在主流的业务架构就是价值流能力,在这套架构体系中,有四个核心元素,分别是价值流、业务能力、信息、组织。 这四个核心元素,特别是对于价值流以及业务能力…

查看主机的php参数short_open_tag 是否为 on

我想要查看主机的php参数short_open_tag 是否为 on,由于我使用的是Hostease的Linux虚拟主机产品,在cPanel面板中并没有找到这个参数选项,因此无法查看。这边联系了Hostease技术支持了解,可以通过以下方式进行查看。 1.先登陆cPane…

Shell编程之条件判断语句

目录 一、条件判断 1、test命令 2、文件测试 3、整数值比较 4、字符串判断 5、逻辑测试 二、if语句 1、if单分支语句 2、双分支语句 3、多分之语句 4、case 分支语句 一、条件判断 Shell环境根据命令执行后的返回状态值(echo $?)来判断是否执行成…

鸿蒙HarmonyOS开发中的易混点归纳-持续补充中

相关文章目录 鸿蒙HarmonyOS开发术语全解:小白也能看懂! 文章目录 相关文章目录前言一、build()函数和Builder装饰器?二、自定义组件和系统组件(内置组件)三、组件和页面四、自定义弹窗和其他弹窗总结 前言 一、build…

SpringBoot(五)之新增注解

SpringBoot(五)之新增注解 文章目录 SpringBoot(五)之新增注解核心注解EnableAutoConfiguration**SpringBootApplication** 条件注解Spring原生Conditional注解手写Conditional注解**ConditionalOnProperty****ConditionalOnMissi…

Logrus IT的供应商经理如何找到优秀地翻译人员

在现代世界,翻译和本地化可以使企业打入新市场并与来自不同国家的客户互动,从而对企业的成功起到决定性作用。翻译过程中的一个关键部分是寻找和招聘翻译。在Logrus IT,这是Anna Gulenkova的工作,我们决定和她谈谈她是如何做到的。…

BUUCTF---misc---我吃三明治

1、下载附件是一张图片 2、在winhex分析,看到一串整齐的编码有点可疑,保存下来,拿去解码,发现解不了,看来思路不对 3、再仔细往下看的时候也发现了一处这样的编码,但是这次编码后面多了一段base编码 4、拿去…

最长前缀数组

欢迎关注更多精彩 关注我,学习常用算法与数据结构,一题多解,降维打击。 基本问题 给定一个字符串s, 返回一个数组Z, Z[i]代表子串s[i…n] 与s最长公共前缀的长度。 朴素做法 可以枚举所有s[i…n]子串,然后与s一一比较&#x…

ERROR TypeError: Cannot read property ‘tapPromise‘ of undefined

问题概要 vue项目运行npm run build打包时,报了ERROR TypeError: Cannot read property ‘tapPromise’ of undefined TypeError: Cannot read property ‘tapPromise’ of undefined的错误。 解决方式 根据github中compression-webpack-plugin源码issues里的提示 …

C#--SVG矢量图画法示例

1.代码示例 <Viewbox Grid.Column"1" Grid.ColumnSpan"1" Grid.RowSpan"1" ><Path Name"ValveShape" Stroke"Black" Data"M 50,0 L 150,200 L 50,200 L 150,0 Z" Width"200" Height"…

经常出差用哪些办公软件记录工作?可多设备同步使用的便签笔记软件

对于许多职场人士来说&#xff0c;出差已成为工作常态。在旅途中&#xff0c;如何高效处理工作&#xff0c;确保信息不遗漏&#xff0c;成为了一个不小的挑战。那么&#xff0c;对于经常需要移动办公的我们&#xff0c;哪款办公软件才是最佳选择呢&#xff1f;可多设备同步使用…

5.23 学习总结

一.项目优化&#xff08;语音通话&#xff09; 实现步骤&#xff1a; 1.用户发送通话申请&#xff0c;并处理通话请求&#xff0c;如果同意&#xff0c;为两个用户之间进行连接。 2.获取到电脑的麦克风和扬声器&#xff0c;将获取到的语音信息转换成以字节数组的形式传递。 …

DataBinding viewBinding(视图绑定与数据双向绑定)简单案例 (kotlin)

先上效果&#xff1a; 4个view的文字都是通过DataBinding填充的。交互事件&#xff1a;点击图片&#xff0c;切换图片 创建项目&#xff08;android Studio 2023.3.1&#xff09; Build.gradle(:app) 引入依赖库&#xff08;完整源码&#xff09; buildFeatures { vie…

java中的HashSet类

一、HashSet类 实现了Set接口&#xff0c;无法存储重复元素 特点&#xff1a;元素位置无序、无索引、底层是HashMap 1、构造方法 内部是HashMap的构造方法 2、add方法 (1)元素在底层存储使用到了三种数据结构&#xff1a;hash数组、链表、树 (2)添加流程&#xff08;根据…

告别登录烦恼,WPS免登录修改器体验!(如何实现不登录使用WPS)

文章目录 &#x1f4d6; 介绍 &#x1f4d6;&#x1f3e1; 演示环境 &#x1f3e1;&#x1f4d2; 解决方案 &#x1f4d2;&#x1f388; 获取方式 &#x1f388;⚓️ 相关链接 ⚓️ &#x1f4d6; 介绍 &#x1f4d6; 想象一下&#xff0c;如果你能够绕过繁琐的登录流程&#x…