数据结构(4)——带哨兵位循环双向链表

目录

前言

一、带哨兵的循环双向链表是什么

二、链表的实现

2.1规定结构体

2.2创建节点

2.3初始化

2.4打印

2.5检验是否为空

2.6销毁链表

2.7尾插

2.8尾删

2.9头插

2.10头删

2.11寻找特定节点

2.12任意位置插入(pos前)

2.13删除任意节点

总结


前言

前面我们学习了单链表的一些知识,由单链表引申出双向链表,同时带哨兵位或者不带哨兵位是两种,但大差不差,这里学习一下带哨兵位的循环双向链表。

其实有很多链表的结构,组成它们的也就是循环非循环,带哨兵位不带哨兵位,双向还是单向。


一、带哨兵的循环双向链表是什么

无哨兵单向非循环链表:结构简单,也就是我们常说的单链表,一般不会用来单独存数据,实际中更多是作为其它数据的子结构,如哈希桶,图的邻接表等等。

带哨兵双向循环链表:结构最复杂,一般用来单独存储数据,实际中使用的链表数据结构,都是带哨兵(头)双向链表,另外这个结构虽然复杂,但是使用代码实现以后会发现结构带来很多优势,实现反而简单了。

二、链表的实现

这里我们要实现的有

//创建节点
LTNode* BuyListNode(LTDataType x)
//初始化
LTNode* LTInit();
//打印
void LTPrint(LTNode* phead);
//销毁链表
void LTDestory(LTNode* phead);
//检验是否为空
bool LTEmpty(LTNode* phead)
//尾插尾删
void LTPushBack(LTNode* phead, LTDataType x);
void LTPopBack(LTNode* phead);
//检验链表是否为空
bool LTEmpty(LTNode* phead);
//头插头删
void LTPushFront(LTNode* phead, LTDataType x);
void LTPopFront(LTNode* phead);
//在pos之前加入一个值
void LTInsert(LTNode* pos, LTDataType x);
//删除节点
void LTErase(LTNode* pos);
//寻找特定节点
LTNode* LTFind(LTNode* phead, LTDataType x);

2.1规定结构体

要想实现一个链表,就要首先规定一下每一个节点的结构体组成部分,这里我们先使用结构体来定义一下。

每一个节点内包括数据域,头指针和尾指针。

typedef int LTDataType;typedef struct ListNode
{struct ListNode* next;//头指针struct ListNode* prev;//尾指针LTDataType data;//数据
}LTNode;

基于这个结构,我们就可以实现后期的一系列操作内容,包括哨兵位的创建。

2.2创建节点

这里我们命名为BuyListNode,返回类型为结构体也就是LTNode*,通过传入一个数据LTDataType x,从而实现对节点的创建。

LTNode* BuyListNode(LTDataType x)
{LTNode* node = (LTNode*)malloc(sizeof(LTNode));if (node == NULL){perror("malloc fail");return NULL;}node->next = NULL;node->prev = NULL;node->data = x;return node;
}

通过malloc分配一个节点的内存,然后把头指针尾指针都赋为空,因为这里不知道头尾指针指向谁,把传入数据,最后返回节点。

2.3初始化

初始化就是初始化一个哨兵位节点,也就是一个头,这里把哨兵位数据定义为-1,头尾指针指向自己,因为这里是双向循环链表,返回此节点。

//初始化
LTNode* LTInit()
{LTNode* phead = BuyListNode(-1);phead->next = phead;phead->prev = phead;return phead;
}

2.4打印

打印这里就是不断访问当前节点的下一个节点,之后打印此节点的数据。

//打印
void LTPrint(LTNode* phead)
{assert(phead);LTNode* cur = phead->next;printf("head<=>");while (cur != phead){printf("%d<=>", cur->data);cur = cur->next;}
}

这里为了形象,打印出来的后面会接上一个<<=>>,来代表双向循环链表。

2.5检验是否为空

 通过一个布尔值来检验是否为空,主要就是检查哨兵位有没有。

//检验是否为空
bool LTEmpty(LTNode* phead)
{assert(phead);return phead->next == phead;
}

2.6销毁链表

我们用 LTDestory来作为销毁链表的函数名字。实际上销毁链表就是先把除了哨兵位(头节点)以外的所有节点删除释放,最后再释放哨兵位,实现是这么实现的:

//销毁释放
void LTDestory(LTNode* phead)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){LTNode* next = cur->next;free(cur);cur = next;}free(phead);phead = NULL;
}

先检查是否有节点,然后把当前cur节点定义为哨兵位的下一个节点,如果cur不为哨兵位(因为是循环链表,所以最后肯定有一个时间,它的尾节点一定指向自己),在循环里面,先用next节点记住cur当前节点的下一个节点,然后释放掉当前节点,再把之前定义的next赋给当前节点,就相当于cur不断再往后走,不断再释放,最后到了循环结束的条件后,释放一下哨兵位就可以实现全部释放。

2.7尾插

这个尾插实现就基于之前创建节点的函数BuyListNode,先创建节点,找到尾节点后,再进行修改。

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = BuyListNode(x);LTNode* tail = phead->prev;//找尾节点tail->next = newnode;newnode->prev = tail;newnode->next = phead;phead->prev = newnode;}

用newnode来代表要插入的新节点,找到尾节点,因为哨兵位的头指针指向最后一个节点,这时候把新的newnode插入到尾节点后面就可以,注意此时新的节点成为了尾节点,所以要更新一下头尾指针的指向关系。

2.8尾删

尾删就是找到当前尾节点,然后把尾节点的上一个节点作为最后一个节点,更新当前节点的头尾指针,删除当前尾节点。

//尾删
void LTPopBack(LTNode* phead)
{assert(phead);assert(!LTEmpty(phead));LTNode* tail = phead->prev;LTNode* tailPrev = tail->prev;tailPrev->next = phead;phead->prev = tailPrev;free(tail);tail = NULL;}

当然这里也要断言一下,看看是否为空,为空就不需要删除。

2.9头插

头插实际上就是在哨兵位的下一个节点插入,实现过程就是类似链表的插入一样。

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = BuyListNode(x);phead->next->prev = newnode;newnode->next = phead->next;phead->next = newnode;newnode->prev = phead;}

插入进去后更新前后指针的指向就可以。

2.10头删

头删就是指定哨兵位的下一个节点,然后这个节点的下一个节点的头指针更新指向哨兵位,哨兵位的下一节点指向它,就然后释放掉刚才的头节点可以。

//头删
void LTPopFront(LTNode* phead)
{assert(phead);LTNode* destorynode = phead->next;phead->next->next->prev = phead;phead->next = phead->next->next;free(destorynode);destorynode = NULL;
}

上面给出了头删的代码。

2.11寻找特定节点

寻找特定节点,就是通过数据x寻找,之后返回一个节点的地址,这里就是通过一个循环寻找的特定节点,然后返回。

//寻找特定节点
LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);//带头双向循环是定不为空的LTNode* cur = phead->next;while (cur != phead){if (cur->data == x){return cur;}cur = cur->next;}return NULL;
}

2.12任意位置插入(pos前)

基于之前的寻找特定的节点,以及新节点的创建,在这里对新节点进行插入,插入到pos的下一个节点,并且更新指针。

//任意位置插入
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* prev = pos->prev;LTNode* newnode = BuyListNode(x);prev->next = newnode;newnode->prev = prev;newnode->next = pos;pos->prev = newnode;
}

2.13删除任意节点

通过传入一个pos节点,进行删除,逻辑和前面的删除差不多。

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

总结

这里对循环有哨兵位双链表进行基本功能的编写和学习。

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

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

相关文章

Github 2025-03-30 php开源项目日报 Top10

根据Github Trendings的统计,今日(2025-03-30统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量PHP项目10TypeScript项目1Coolify: 开源自助云平台 创建周期:1112 天开发语言:PHP, Blade协议类型:Apache License 2.0Star数量:10527 个Fo…

3. 线程间共享数据

1. 线程共享数据会造成什么问题&#xff1f; 1.1 读写不一致 多线程读不会造成数据变动&#xff0c;所以没有问题。只要有一个线程设计修改数据&#xff0c;就会导致数据共享出现问题&#xff0c;简单的是数据不一致&#xff0c;严重的是程序访问已经释放的内存&#xff0c;造…

JAVA垃圾回收算法和判断垃圾的算法

一、判断垃圾的算法 判断对象是否为垃圾的核心是确定对象是否不再被使用。Java主要采用以下两种算法&#xff1a; 1. 引用计数法&#xff08;Reference Counting&#xff09; 原理&#xff1a;每个对象维护一个引用计数器&#xff0c;记录被引用的次数。当引用被添加时计数器…

界面架构 - MVVM (Qt)

MVVM MVVM 的主要特点示例示例功能示例代码ViewModel 类&#xff08;C&#xff09;主函数入口&#xff08;main.cpp&#xff09; QML 文件&#xff08;main.qml&#xff09;总结 MVVM&#xff08;Model-View-ViewModel&#xff09;架构是一种旨在进一步分离界面和业务逻辑的设计…

第十四届MathorCup高校数学建模挑战赛-C题:基于 LSTM-ARIMA 和整数规划的货量预测与人员排班模型

目录 摘要 一、 问题重述 1.1 背景知识 1.2 问题描述 二、 问题分析 2.1 对问题一的分析 2.2 对问题二的分析 2.3 对问题三的分析 2.4 对问题四的分析 三、 模型假设 四、 符号说明 五、 问题一模型的建立与求解 5.1 数据预处理 5.2 基于 LSTM 的日货量预测模型 5.3 日货量预测…

银河麒麟V10 aarch64架构安装mysql教程

国产操作系统 ky10.aarch64 因为是arm架构&#xff0c;故选择mysql8&#xff0c;推荐安装8.0.28版本 尝试8.0.30和8.0.41版本均未成功&#xff0c;原因不明☹️ 1. 准备工作 ⏬ 下载地址&#xff1a;https://downloads.mysql.com/archives/community/ 2. 清理历史环境 不用管…

C++多继承

可以用多个基类来派生一个类。 格式为&#xff1a; class 类名:类名1,…, 类名n { private: … &#xff1b; //私有成员说明; public: … &#xff1b; //公有成员说明; protected: … &#xff1b; //保护的成员说明; }; class D: public A, protected B, private C { …//派…

某地老旧房屋自动化监测项目

1. 项目简介 自从上个世纪90年代以来&#xff0c;我国经济发展迅猛&#xff0c;在此期间大量建筑平地而起&#xff0c;并且多为砖混结构的住房&#xff0c;使用寿命通常约为30-50年&#xff0c;钢筋混凝土结构&#xff0c;钢结构等高层建筑&#xff0c;这些建筑在一般情况下的…

产品经理的大语言模型课 04 -模型应用的云、边、端模式对比

目录 算力部署方式的影响因素数据量计算难度前期投入数据隐私应用规模与泛化能力 云、边、端部署的特点和对比典型场景举例社区人脸门禁后厨老鼠识别 未来展望 算力部署方式的影响因素 最近和人工智能从业者进行了非常广泛的沟通&#xff0c;尝试对模型应用的云、边、端模式进…

基于Python设计的TEQC数据质量可视化分析软件

标题:基于Python设计的TEQC数据质量可视化分析软件 内容:1.摘要 本文旨在设计一款基于Python的TEQC数据质量可视化分析软件。随着全球导航卫星系统&#xff08;GNSS&#xff09;的广泛应用&#xff0c;数据质量的评估变得至关重要。TEQC&#xff08;TransEditQualityCheck&…

Flinksql--订单宽表

参考&#xff1a; https://chbxw.blog.csdn.net/article/details/115078261 (datastream 实现) 一、ODS 模拟订单表及订单明细表 CREATE TABLE orders (order_id STRING,user_id STRING,order_time TIMESTAMP(3),-- 定义事件时间及 Watermark&#xff08;允许5秒乱序&#x…

粒子滤波介绍

目录 粒子滤波的主要流程可以分为以下 5 个步骤&#xff1a; 粒子滤波&#xff08;PF&#xff09; vs. ESKF&#xff08;误差状态卡尔曼滤波&#xff09; 粒子滤波的主要流程可以分为以下 5 个步骤&#xff1a; 初始化&#xff08;Initialization&#xff09; 生成 N 个粒子&…

一场国际安全厂商的交流会议简记

今天参与了一场国际安全厂商A公司组织的交流会议 与会有国际TOP企业跨境企业 还有国内一些头部商业公司。 A公司很有意思介绍了自己是怎么做安全运营中心SOC的。 介绍了很多内容&#xff0c;包括他们自己的员工量/设备量/事件量/SOC中心人员量&#xff0c;其中人员量只有个位数…

Java面试黄金宝典30

1. 请详细列举 30 条常用 SQL 优化方法 定义 SQL 优化是指通过对 SQL 语句、数据库表结构、索引等进行调整和改进&#xff0c;以提高 SQL 查询的执行效率&#xff0c;减少系统资源消耗&#xff0c;提升数据库整体性能的一系列操作。 要点 从索引运用、查询语句结构优化、数据…

花洒洗澡完毕并关闭后过段时间会突然滴水的原因探究

洗澡完毕后的残留水 在洗澡的过程中&#xff0c;我们通常会使用到大量的水。这些水会通过花洒管子到达花洒顶喷流出。由于大顶喷花洒的喷头较大&#xff0c;关闭后里面的存水会更多。 气压失衡后的滴水 当花洒关闭后&#xff0c;内部的水管和花洒头中仍存有一定量的水。由于…

QSettings用法实战(相机配置文件的写入和读取)

很多情况&#xff0c;在做项目开发的时候&#xff0c;将参数独立出来是比较好的方法 例如&#xff1a;相机的曝光次数、曝光时长等参数&#xff0c;独立成ini文件&#xff0c;用户可以在外面修改即可生效&#xff0c;无需在动代码重新编译等工作 QSettings便可以实现该功能 内…

运维培训班之最佳选择(The best Choice for Operation and Maintenance Training Courses)

运维培训班之最佳选择 从面试官的角度聊聊培训班对运维的帮助&#xff0c;同时给培训班出身的运维一些建议~ 谈到运维&#xff08;尤其是零基础非科班转行的运维&#xff09;找工作&#xff0c;培训班是个不可回避的讨论热点。虽然本人也做过兼职运维培训老师&#xff0c;多少…

网络安全与防护策略

随着信息技术的飞速发展&#xff0c;互联网已成为现代社会不可或缺的一部分。从日常生活到企业运营&#xff0c;几乎所有活动都离不开网络。然而&#xff0c;网络的开放性和广泛性也使得网络安全问题愈发严峻。无论是个人数据泄露&#xff0c;还是大规模的网络攻击&#xff0c;…

LLM 分词器Tokenizer 如何从 0 到 1 训练出来

写在前面 大型语言模型(LLM)处理的是人类的自然语言,但计算机本质上只能理解数字。Tokenizer(分词器) 就是架在自然语言和计算机数字表示之间的一座至关重要的桥梁。它负责将我们输入的文本字符串分解成模型能够理解的最小单元——Token,并将这些 Token 转换成对应的数字…

【ArcGIS微课1000例】0142:如何从谷歌地球保存高清影像图片

文章目录 一、选取影像区域1. 搜索地图区域2. 导入矢量范围二、添加输出图层三、保存高清影像1. 地图选项2. 输出分辨率3. 保存图像四、注意事项一、选取影像区域 首先需要选取影像区域,可通过以下方式快速定位。 1. 搜索地图区域 在搜索框内输入关键词,例如青海湖,点击【…