带头双向循环链表实现

1.结构及特性

前面我们实现了无头单向非循环链表,它的结构是这样的:

在这里的head只是一个指向头结点的指针,而不是带头链表的头节点。

而带头双向循环链表的逻辑结构则是这样的

这就是链表的结构,链表的每一个节点都有两个指针,一个 prev 指针指向前一个节点,而 next 指针则指向下一个节点,而当链表为空时,我们就只有一个哨兵位的头节点,

此时头结点的 prev 和 next 都是指向自己的,如果将prev和next设置为空的话,就不符合循环的结构了。带头双向循环链表相对于无头单向非循环链表的优点就是以下几个

1.有哨兵位的头,这时候我们插入数据的时候就不用分情况讨论,因为无论是第一个插入数据还是正常插入数据都是修改的 头节点 的next ,而不会修改头节点的指针,这样一来我们也不用传二级指针来操作了。我们只需要传头结点的指针 phead ,而后我们的插入删除数据都是对 头节点的 prev和next 进行操作,对结构体成员进行修改只要传结构体指针就行了。

2.双向链表,双向链表在进行指定位置之前插入和删除的时候就不用遍历链表去找指定位置的前一个节点了,我们直接用指定节点的prev就能找到前一个节点。

3.循环,循环链表的特性就是头节点的prev指向尾节点,尾节点的next指向头节点,这样一来,对链表进行尾插尾删操作时我们也不用遍历链表去找尾节点和前一个结点了,我们通过头节点就能删除和插入尾节点。

带头双向循环链表虽然结构比单链表复杂得多,但是它的结构带来了很多的优势,对于各种操作都很方便,完美解决了顺序表的所有问题。

2.链表的接口实现

首先我们定义结点的结构,一个数据和两个指向同类型节点的指针。

typedef int LTDataType;typedef struct ListNode
{struct ListNode* prev;LTDataType data;struct ListNode* next;
}LTNode;

链表初始化

在这个链表结构的实现中,我们首先要在主函数内定义一个头结点,然后对头节点进行初始化。

//初始化
void LTInit(LTNode* phead)
{assert(phead);phead->prev = phead;phead->next = phead;
}

由于头节点的数据我们是不会用的,所以我们直接不管他的数据,将两个指针都指向自己就行了

链表的打印

为了后续的测试方便,我们先将打印函数写出来,打印链表是要遍历链表的,而循环链表的尾节点不会指向NULL,而是指向头节点,所以我们遍历的循环结束条件就是 cur->next != phead ,遍历也是从head的next开始的。同时为了展现双向链表的特性,我们在开头和最后都打印一盒phead,同时数据之间用<=>来连接。

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

 我们先来测试一下空链表的打印

void test1()
{LTNode head;LTInit(&head);LTPrint(&head);}

 

链表销毁

链表的销毁也是从头结点的 next 开始遍历销毁,因为我们的头节点 head 不是动态开辟出来的,所以千万不要对 phead 进行 free 操作,而且我们的头节点也不需要 free ,当链表销毁之后我们对phead的prev和next进行置空。

//销毁链表
void LTDestroy(LTNode* phead)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){LTNode* del = cur;cur = cur->next;free(del);}phead->prev = NULL;phead->next = NULL;
}

创建节点

由于我们插入数据有多个接口,所以我们写一个创建节点的函数出来。


//创建节点
LTNode* NewLTNode(LTDataType x)
{LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));if (newnode == NULL){perror("malloc fail");exit(-1);}newnode->data = x;newnode->prev = NULL;newnode->next = NULL;return newnode;
}

头插 

头插数据在无头单链表的实现中要分两种情况,而在带头链表中我们就都可以一视同仁了。我们正常头插时,首先创建一个新节点,然后让新节点的next指向原来的第一个数据节点,再让原来的第一个数据节点的prev指向新节点,处理完新节点和原头节点的连接关系后再来处理新节点与哨兵位的关系,让新节点的prev指向phead,phead的next指向新节点。

这里同文字描述起来很复杂,但是用图和代码解释起来就会很简单,因为插入节点就只是改变了几个指针的指向。

我们也可以看一下空链表插入是否需要单独讨论。因为空链表的phead 的next和prev都指向phead自己,按照上面的逻辑,首先让新节点的 next 指向phead的 next ,也就是新节点的next 指向phead,然后让phead 的next 的prev指向新节点,也就是phead的prev指向新节点。然后再让新节点的prev指向phead,再让phead的next指向新节点。

文字描述还是很复杂,用途来说明就能知道同样的逻辑也适用于空链表头插。

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

写完头插代码之后我们测试一下头插。

	LTNode head;LTInit(&head);LTPushFront(&head, 1);LTPushFront(&head, 2);LTPushFront(&head, 3);LTPushFront(&head, 4);LTPrint(&head);

尾插

尾插的方式跟头插类似,主要也是要注意指针修改的顺序,要先对新节点和原来的尾节点进行链接,然后再对phead和新节点链接。

其中1和2的顺序是可以变的,3和4 的顺序也是可以变的。再来看一下空链表的尾插。

我们可以发现他们的逻辑是一样的,所以也不要单独处理空链表的尾插。

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = NewLTNode(x);newnode->prev = phead->prev;phead->prev->next = newnode;newnode->next = phead;phead->prev = newnode;
}

尾插测试

//尾插尾删
void test2()
{LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTPrint(&head);}

头删

头删我们只需要把phead的next指向第二个数据节点,而后让第二个数据节点的prev指向phead,最后释放第一个节点就行了。我们再看一下只有一个数据时的头删要不要单独处理,首先将phead的next指向phead的next 的next,也就是phead自己,再将phead的next的next 也就是phead的prev指向phead,然后释放掉数据节点,这时候phead就回到了初始化的样子,所以上面的逻辑也完全适用于一个节点的头删.

但是在删除数据之前我们要判断是否为空链表,为了方便,因为后面的尾删也需要判断,所以我们写一个函数来实现判断。

//判断链表是否为空
bool LTEmpty(LTNode* phead)
{assert(phead);//如果为空链表就返回true//否则返回falsereturn phead->next == phead;
}
//头删
void LTPopFront(LTNode* phead)
{assert(phead);assert(!LTEmpty(phead));//链表为空LTNode* del = phead->next;phead->next = phead->next->next;phead->next->next->prev = phead;free(del);
}

测试头删

一个一个把数据删完

	LTNode head;LTInit(&head);LTPushFront(&head, 1);LTPushFront(&head, 2);LTPushFront(&head, 3);LTPushFront(&head, 4);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);

再测试对空链表头删

	LTNode head;LTInit(&head);LTPushFront(&head, 1);LTPushFront(&head, 2);LTPushFront(&head, 3);LTPushFront(&head, 4);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);LTPrint(&head);LTPopFront(&head);

尾删

尾删函数我们也是要找到倒数第二个节点,然后将这个节点和phead链接起来,最后free尾节点就行了。

当只有一个节点的时候尾删,

他们的逻辑也是一样的

//尾删
void LTPopBack(LTNode* phead)
{assert(phead);assert(!LTEmpty(phead));//判断是否为空链表LTNode* del = phead->prev;phead->prev->prev->next = phead;//倒数第二个节点链接pheadphead->prev = phead->prev->prev;free(del);
}

测试尾删,先逐个删除直到删空

	LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTPrint(&head);LTPopBack(&head);LTPrint(&head);LTPopBack(&head);LTPrint(&head);LTPopBack(&head);LTPrint(&head);LTPopBack(&head);LTPrint(&head);

再对空链表尾删

查找函数

查找函数只能便利查找,遍历的循环条件也是和打印函数一样的,找到就返回链表节点指针,找不到就返回空指针。

//查找
LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){if (cur->data == x){return cur;}cur = cur->next;}//找不到返回NULLreturn NULL;
}

我们测试查找函数直接同时兼顾修改函数的功能。

LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTNode* pos = LTFind(&head,1);pos->data = 10;LTPrint(&head);

这时候我们通过返回的结点指针将1修改成了10,说明函数功能是正常的。

指定位置(之前)插入

我们在指定位置之前插入数据只需要将这个节点的前一个结点与新节点链接起来,再将新节点与该节点链接起来就行了。

当pos是第一个数据节点的时候,也就是头插的时候,逻辑与上面的也是一样的

当pos是phead,这时候在哨兵头节点之前插入就是尾插了,逻辑也和正常插入一样 

我们实现插入函数的时候就不用考虑空链表的情况了,这是使用者要考虑的事情,然后我们只要判断pos是否为NULL就行了。

//指定位置之前处插入
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* newnode = NewLTNode(x);pos->prev->next = newnode;newnode->prev = pos->prev;pos->prev = newnode;newnode->next = pos;}

测试插入函数

	LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTNode* pos = LTFind(&head,1);LTInsert(pos, 10);LTPrint(&head);

这说明我们的插入函数是能够成功实现头插的。

然后再测试是否能够实现尾插

	LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTNode* pos = LTFind(&head,1);LTInsert(pos, 10);LTInsert(&head, 50);LTPrint(&head);

功能正常,这时候我们就能把前面写的头插尾插函数修改成用插入函数来实现的了。

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTInsert(phead->next, x);
}//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTInsert(phead, x);
}

删除指定位置

在这个函数中我们不用考虑链表为空的情况,这也是使用者该注意的事情。而指定位置的删除首先就是要将指定位置的前一个结点和后一个节点链接起来,然后再free掉pos节点。

而这个逻辑对于头节点和尾节点肯定也是适用的,大家可以自己去画图验证了。

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

我们来测试一下删除头节点的情况

	LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTErase(head.next);LTPrint(&head);

再测试一下删除尾节点

	LTNode head;LTInit(&head);LTPushBack(&head, 1);LTPushBack(&head, 2);LTPushBack(&head, 3);LTPushBack(&head, 4);LTErase(head.prev);LTPrint(&head);

函数功能实现地没有问题,这时候我们就能用Erase去改造之前写的头删尾删了。

//头删
void LTPopFront(LTNode* phead)
{assert(phead);//头删要判断链表是否为空assert(!LTEmpty(phead));LTErase(phead->next);
}//尾删
void LTPopBack(LTNode* phead)
{assert(phead);//尾删也要判断链表是否为空assert(!LTEmpty(phead));LTErase(phead->prev);
}

求数据个数

我们可以再设计一个函数来求链表的数据个数,同样是要遍历链表,与打印函数的遍历是一样的。

//求数据个数
size_t LTSize(LTNode* phead)
{assert(phead);if(LTEmpty(phead)){return 0;}size_t size = 0;LTNode* cur = phead->next;while (cur!=phead){size++;cur = cur->next;}return size;
}

总结

虽然带头双向循环链表的结构比单链表要复杂的多,但是他的结构带来的优势就是增删的操作都十分简单,而且头插尾插能完全复用插入函数,头删尾删也能完全复用删除函数,代码量变小了很多,只要我们理解了指针之间的逻辑关系,这种结构使用起来就十分方便了。

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

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

相关文章

docker 安装redis报错:can not init background jbos

启动redis&#xff0c;发现一直再重启 docker run -d --name redis -p 6379:6379 --restartalways redis:6.2.6 --requirepass "123456" 查看日志&#xff0c;发现job没启动 docker logs 47f6572a779c 尝试了一堆解决办法。。。最后发现尝试安装了redis6.2.6版本&a…

从《布瓦尔与佩库歇》实践中学习社会科学概论

从《布瓦尔与佩库歇》实践中学习社会科学概论 前情提要《布瓦尔与佩库歇》实践笔记云藏山鹰社会科学概论报告核心--信息形数身知™意合™意气实体过程意气实体过程宇宙学诠释™ 社会科学概论花间流风版导读&#xff0c;马斯克风格演讲[ 一尚韬竹团队供稿&#xff1b;] 内容展开…

实验4 层次图和HIPO图

一、实验目的 通过绘制层次图和HIPO图&#xff0c;熟练掌握层次图和HIPO图的基本原理。 能对简单问题进行层次图和HIPO图的分析&#xff0c;独立地完成层次图和HIPO图设计。 二、实验项目内容&#xff08;实验题目&#xff09; 1、用Microsoft Visio绘制出图书馆管理系统的层…

Flume 拦截器概念及自定义拦截器的运用

文章目录 Flume 拦截器拦截器的作用拦截器运用1.创建项目2.实现拦截器接口3.编写事件处理逻辑4.拦截器构建5.打包与上传6.编写配置文件7.测试运行 Flume 拦截器 在 Flume 中&#xff0c;拦截器&#xff08;Interceptors&#xff09;是一种可以在事件传输过程中拦截、处理和修改…

Spring定义Bean对象笔记(二)

前言&#xff1a;上一篇记录了通过XML文件来定义Bean对象&#xff0c;这一篇将记录通过注解和配置类的方式来定义Bean对象。 核心注解&#xff1a; 定义对象&#xff1a;Component,Service,Repository,Controller 依赖注入&#xff1a; 按类型&#xff1a;Autowired 按名称&am…

【JavaScript】作用域 ③ ( JavaScript 作用域链 | 作用域链变量查找机制 )

文章目录 一、JavaScript 作用域链1、作用域2、作用域链3、作用域链变量查找机制 二、代码示例 - 作用域链 一、JavaScript 作用域链 1、作用域 在 JavaScript 中 , 任何代码都有 作用域 , 全局作用域 : 在 <script> 标签中 或者 js 脚本中 定义的变量 属于 全局作用域 …

Vue3【进阶】

简介 https://cn.vuejs.org/guide/introduction.html 创建vue3工程 【基于 vue-cli创建】 基本和vue-cli的过程类似&#xff0c;只是选择的时候用vue3创建 【基于vite创建】【推荐】 【官网】https://vitejs.cn/ 【可以先去学一下webpack】 步骤 【https://cn.vitejs.…

kubernetes集群添加到jumpserver堡垒机里管理

第一步、在kubernetes集群中获取一个永久的token。 jumpserver堡垒机用api的来管理kubernetes&#xff0c;所以需要kubernetes的token&#xff0c;这个token还需要是一个永久的token&#xff0c;版本变更&#xff1a;Kubernetes 1.24基于安全方面的考虑&#xff08;特性门控Le…

LeetCode-热题100:118. 杨辉三角

题目描述 给定一个非负整数 numRows&#xff0c;生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中&#xff0c;每个数是它左上方和右上方的数的和。 示例 1: 输入: numRows 5 输出: [[1],[1,1],[1,2,1],[1,3,3,1],[1,4,6,4,1]] 示例 2: 输入: numRows 1 输出: [[1]]…

代码随想录第32天|455.分发饼干 376. 摆动序列

理论基础 贪心算法核心&#xff1a;选择每一阶段的局部最优&#xff0c;从而达到全局最优。 455.分发饼干 455. 分发饼干 - 力扣&#xff08;LeetCode&#xff09;代码随想录 (programmercarl.com)455. 分发饼干 - 力扣&#xff08;LeetCode&#xff09; 贪心算法理论基础&am…

【AI绘画/作图】风景背景类关键词模板参考

因为ds官网被墙,所以翻了IDE的源码整理了下stablestudio里的官方模板&#xff0c;顺便每个模板生成了一份…不知道怎么写关键词的可以参考 Stunning sunset over a futuristic city, with towering skyscrapers and flying vehicles, golden hour lighting and dramatic cloud…

C语言高效的网络爬虫:实现对新闻网站的全面爬取

1. 背景 搜狐是一个拥有丰富新闻内容的网站&#xff0c;我们希望能够通过网络爬虫系统&#xff0c;将其各类新闻内容进行全面地获取和分析。为了实现这一目标&#xff0c;我们将采用C语言编写网络爬虫程序&#xff0c;通过该程序实现对 news.sohu.com 的自动化访问和数据提取。…

行业型软文怎么写,媒介盒子分享

行业型软文即指面对行业内人群的软文,此类文章的目的通常是为了扩大行业影响力,奠定行业品牌地位。企业的行业地位将直接影响到其核心竞争力,甚至会影响到最终用户的选择。今天媒介盒子就和大家聊聊行业型软文怎么写。 一、了解受众需求&#xff1a; 首先&#xff0c;深入研究…

【C++第三阶段】string容器

以下内容仅为当前认识&#xff0c;可能有不足之处&#xff0c;欢迎讨论&#xff01; 文章目录 string容器基本概念构造函数赋值操作拼接操作字符串查找和替换字符串比较字符串存取字符串插入和删除字符串子串 string容器 基本概念 本质&#x1f449;string是C风格的字符串&…

快速获取文件夹及其子文件夹下的所有文件名

1、在文件夹中新建文本文档&#xff0c;命名为“命令.txt” 2、输入以下内容 tree /F > 文件名.txt dir *.* /B > 文件名.txt 其中文件名和文件格式可以是任意的&#xff0c;tree命令可生成文件及其子文件夹下所有文件的名称&#xff0c;dir命令只生成当前目…

网络基础三——初识IP协议

网络基础三 ​ 数据通过应用层、传输层将数据传输到了网络层&#xff1b; ​ 传输层协议&#xff0c;如&#xff1a;TCP协议提供可靠性策略或者高效性策略&#xff0c;UDP提供实时性策略&#xff0c;保证向下层交付的数据是符合要求的的&#xff1b;而网络层&#xff0c;如&a…

LeetCode 40. 组合总和 II

解题思路 cand[]{1,2,3,4} target4; 相关代码 class Solution {List<List<Integer>> res;List<Integer> path;boolean st[];public List<List<Integer>> combinationSum2(int[] candidates, int target) {path new ArrayList<>();res …

c++前言

目录 1. 什么是 C 2. C 发展史 3. C 的重要性 4. 如何学习 C 5. 关于本门课程 1. 什么是C C语言是结构化和模块化的语言&#xff0c;适合处理较小规模的程序。对于复杂的问题&#xff0c;规模较大的 程序&#xff0c;需要高度的抽象和建模时&#xff0c; C 语言则不合适…

Acwing.1388 游戏(区间DP对抗思想)

题目 玩家一和玩家二共同玩一个小游戏。 给定一个包含 N个正整数的序列。 由玩家一开始&#xff0c;双方交替行动。 每次行动可以在数列的两端之中任选一个数字将其取走&#xff0c;并给自己增加相应数字的分数。&#xff08;双初始分都是 0分&#xff09; 当所有数字都被…

备战蓝桥杯---线段树应用2

来几个不那么模板的题&#xff1a; 对于删除&#xff0c;我们只要给那个元素附上不可能的值即可&#xff0c;关键问题是怎么处理序号变化的问题。 事实上&#xff0c;当我们知道每一个区间有几个元素&#xff0c;我们就可以确定出它的位置&#xff0c;因此我们可以再维护一下前…