数据结构——lesson4带头双向循环链表实现

2024-02-29 21:04

本文主要是介绍数据结构——lesson4带头双向循环链表实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言✨✨

💥个人主页:大耳朵土土垚-CSDN博客

💥 所属专栏:数据结构学习笔记​​​​​​

💥双链表与单链表的区分:单链表介绍与实现

💥对于malloc函数有疑问的:动态内存函数介绍

   感谢大家的观看与支持🌹🌹🌹 

   有问题可以写在评论区或者私信我哦~

 

目录

前言✨✨

一、💥💥什么是带头双向循环链表?

二、🥳🥳带头双向循环链表的实现 

1 .搭建链表基础

2.从内存中开辟一个节点

3. 创建返回链表的头结点

4.双向链表销毁

5.双向链表打印 

6.双向链表尾插 

7.双向链表尾删

8.双向链表头插 

9.双向链表头删 

10.双向链表查找

11.双向链表在pos的前面进行插入 

12.双向链表删除pos位置的节点 

三、💫💫拓展

四、🎉🎉结言 


一、💥💥什么是带头双向循环链表?

 

带头双向循环链表(Doubly Circular Linked List with a Head)是一种链表数据结构,它具有以下特点:

1.头节点:带头双向循环链表包含一个头节点,它位于链表的起始位置,并且不存储实际数据。头节点的前驱指针指向尾节点,头节点的后继指针指向第一个实际数据节点。

2.循环连接:尾节点的后继指针指向头节点,而头节点的前驱指针指向尾节点,将链表形成一个循环连接的闭环。这样可以使链表在遍历时可以无限循环,方便实现循环操作。

3.双向连接:每个节点都有一个前驱指针和一个后继指针,使得节点可以向前和向后遍历。前驱指针指向前一个节点,后继指针指向后一个节点。

        总结:带头双向循环链表可以支持在链表的任意位置进行插入和删除操作,并且可以实现正向和反向的循环遍历。通过循环连接的特性,链表可以在连续的循环中遍历所有节点,使得链表的操作更加灵活和高效。

如下图所示:

 

 

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

二、🥳🥳带头双向循环链表的实现 

1 .搭建链表基础

带头双向循环链表需要三个变量,两个存放指向前后节点的指针,另一个存放数据

// 带头+双向+循环链表增删查改实现
typedef int LTDataType;
typedef struct ListNode
{LTDataType data;//存放数据struct ListNode* next;//指向下一个节点struct ListNode* prev;//指向上一个节点
}ListNode;

2.从内存中开辟一个节点

使用malloc函数开辟节点

//从内存中开辟一个节点
ListNode* BuyNode(LTDataType x)
{ListNode* buynode = (ListNode*)malloc(sizeof(struct ListNode));if (buynode == NULL)//开辟失败{perror("malloc fail");}buynode->data = x;buynode->next = NULL;buynode->prev = NULL;}

 

3. 创建返回链表的头结点
 

开始时头节点两个指针都指向自己

//创建返回链表的头结点.
ListNode* ListCreate()
{ListNode* head = BuyNode(-1);//这里将头节点数据设为-1,任意数都可以head->next = head;head->prev = head;return head;
}

 

4.双向链表销毁

 malloc开辟空间后要使用free销毁内存空间,防止内存泄漏

// 双向链表销毁
void ListDestory(ListNode* pHead)
{assert(pHead);ListNode* cur = pHead->next;//头节点最后销毁while (cur != pHead)//循环一遍{ListNode* next = cur->next;//保存下一个节点,防止丢失free(cur);//销毁节点cur = next;}free(pHead);//销毁头节点
}

5.双向链表打印 

 

//双向链表打印
void ListPrint(ListNode* pHead)
{assert(pHead);if (pHead->next == pHead)//没有节点的情况,也可以不考虑{printf("pHead<=>pHead");return;}//有节点的情况printf("pHead<=>");//先打印pHeadListNode* cur = pHead->next;while (cur != pHead){printf("%d<=>", cur->data);cur = cur->next;}printf("pHead");//因为最后也是指向pHead
}

 

没有节点情况打印如下: 

6.双向链表尾插 

 

// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{assert(pHead);//找尾节点,保存原来的尾//尾节点就是pHead->prevListNode* tail = pHead->prev;//开辟新节点ListNode* newnode = BuyNode(x);//尾插pHead->prev = newnode;newnode->next = pHead;newnode->prev = tail;tail->next = newnode;}

 

结果如下:

 

7.双向链表尾删

// 双向链表尾删
void ListPopBack(ListNode* pHead)
{assert(pHead);//没有节点不能尾删,头节点pHead不算if (pHead->next == pHead){printf("没有添加节点\n");return;}//找尾节点,以及尾节点的前一个节点ListNode* tail = pHead->prev;ListNode* tailprev = tail->prev;//尾删tailprev->next = pHead;pHead->prev = tailprev;free(tail);//释放内存空间
}

 结果如下:

8.双向链表头插 

// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{assert(pHead);//找头以外的第一个节点ListNode* headnext = pHead->next;//创建新节点ListNode* newnode = BuyNode(x);//头插pHead->next = newnode;newnode->next = headnext;newnode->prev = pHead;headnext->prev = newnode;
}

 

结果如下:

9.双向链表头删 

 

// 双向链表头删
void ListPopFront(ListNode* pHead)
{assert(pHead);//判断有没有节点,头节点pHead除外if (pHead->next == pHead){printf("没有添加节点\n");return;}//有节点//找头节点以及头节点的下一个节点ListNode* head = pHead->next;ListNode* headnext = head->next;//头删pHead->next = headnext;headnext->prev = pHead;free(head);//释放内存空间
}

 

 结果如下:

 

10.双向链表查找

// 双向链表查找
ListNode* ListFind(ListNode* pHead, LTDataType x)
{assert(pHead);//判断有无节点if (pHead->next == pHead){printf("没有添加节点\n");return;}ListNode* cur = pHead->next;//遍历查找while (cur){if (cur->data == x){return cur;//找到返回地址}cur = cur->next;}
}

结果如下:

11.双向链表在pos的前面进行插入 

在pos位置前面插入原理和头插尾插相似

// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{assert(pos);//找到pos前一个节点ListNode* posprev = pos->prev;//创建新节点ListNode* newnode = BuyNode(x);//在pos前插入posprev->next = newnode;newnode->next = pos;newnode->prev = posprev;pos->prev = newnode;}

结果如下:

 

12.双向链表删除pos位置的节点 

在pos位置删除原理和头删尾删相似

// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{assert(pos);//找到pos前一个节点ListNode* posprev = pos->prev;//找打pos后一个节点ListNode* posnext = pos->next;//删除pos位置节点posprev->next = posnext;posnext->prev = posprev;free(pos);//释放内存空间}

结果如下:

三、💫💫拓展

思考:在pos之前插入与头插尾插是否有关?

           在pos位置删除与头删尾删是否相似?

 

我们发现pos位置前插入函数代码似乎可以复用在头插尾插;

pos位置删除函数代码似乎可以复用在头删尾删;

下面我们一起来实现

1.尾插头插 

//尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{assert(pHead);//找尾节点,保存原来的尾//尾节点就是pHead->prev//ListNode* tail = pHead->prev;开辟新节点//ListNode* newnode = BuyNode(x);尾插//pHead->prev = newnode;//newnode->next = pHead;//newnode->prev = tail;//tail->next = newnode;ListInsert(pHead, x);}//头插
void ListPushFront(ListNode* pHead, LTDataType x)
{assert(pHead);//找头以外的第一个节点//ListNode* headnext = pHead->next;创建新节点//ListNode* newnode = BuyNode(x);头插//pHead->next = newnode;//newnode->next = headnext;//newnode->prev = pHead;//headnext->prev = newnode;ListInsert(pHead->next, x);}

2.尾删,头删

 

// 双向链表尾删
void ListPopBack(ListNode* pHead)
{assert(pHead);//没有节点不能尾删,头节点pHead不算if (pHead->next == pHead){printf("没有添加节点\n");return;}找尾节点,以及尾节点的前一个节点//ListNode* tail = pHead->prev;//ListNode* tailprev = tail->prev;尾删//tailprev->next = pHead;//pHead->prev = tailprev;//free(tail);//释放内存空间ListErase(pHead->prev);
}// 双向链表头删
void ListPopFront(ListNode* pHead)
{assert(pHead);//判断有没有节点,头节点pHead除外if (pHead->next == pHead){printf("没有添加节点\n");return;}有节点找头节点以及头节点的下一个节点//ListNode* head = pHead->next;//ListNode* headnext = head->next;头删//pHead->next = headnext;//headnext->prev = pHead;//free(head);//释放内存空间ListErase(pHead->next);
}

 

 运行结果依然不受影响:

 

四、🎉🎉结言 

        我们通过上面的学习发现,相似的代码的重复利用可以大大减少我们写代码的时间与精力,提高我们工作学习的效率;双向链表尽管结构较单链表复杂,但其实现却比单链表简单得多,相信大家对此都深有体会,此外数据结构的题目我们可以通过画图来很好的获得思路与接替步骤,以上就是带头双向循环链表的相关知识啦~完结撒花~🎉🎉🌹🌹🌹

这篇关于数据结构——lesson4带头双向循环链表实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/760078

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja