【数据结构】—带头双向循环链表的实现(完美链表)

简介: 【数据结构】—带头双向循环链表的实现(完美链表)

目录


前言

链表的实现

新节点的创建

链表初始化

尾插与尾删

头插与头删

查找数据

在任意位置的插入与删除

链表的销毁

总结

前言


链表结构一共有八种形式,在前面的文章里已经讲完了不带头单向非循环链表的实现,但是我们发现该链表实现尾插与尾删时比较麻烦,要先从头节点进行遍历,找到尾节点,时间复杂度为O(N),而本次所讲的带头双向循环单链表,则可以直接找到尾节点。


1.png


虽然该链表看起来特别复杂,但实际上真正实现起来很简单,并且用起来真的超爽,还能拿来吹吹牛皮。唬一唬一知半解的外行人。


链表的实现


typedef int LTDataType;//类型重命名
typedef struct ListNode
{
  LTDataType _data;//数据
  struct ListNode* _next;//指向下一个节点的指针
  struct ListNode* _prev;//指向前一个节点的指针
}ListNode;


新节点的创建

这里由于后面的插入都需要进行创建新节点,所以我们把它写成一个函数,后面进行插入操作的时候,直接调用即可。这里没什么技术含量。直接malloc即可


ListNode* BuyListNode(LTDataType x)
{
  ListNode* phead = (ListNode*)malloc(sizeof(ListNode));
  if (phead == NULL)
  {
  perror("malloc fail");
  exit(-1);
  }
  phead->_data = x;
  phead->_next = NULL;
  phead->_prev = NULL;
  return phead;
}


链表初始化

空表状态下应该是如下图这样的,因为它是带头的循环链表,所以第一个节点不用来存储有效数据。它的next与prev都指向自己就说明该链表是空表。


2.png


ListNode* InitListNode()
{
  ListNode* phead = BuyListNode(-1);//这里的-1不是有效数据
  phead->_next = phead;
  phead->_prev = phead;
  return phead;
}


尾插与尾删

尾插


尾插首先要找到尾节点,这里的尾节点很容易找到,就是头节点的prev指向的节点。如下:


3.gif


这里的尾插也满足空表情况下进行尾插。所以该代码没问题


代码


void ListPushBack(ListNode* pHead, LTDataType x)
{
  assert(pHead);
  ListNode* newnode = BuyListNode(x);//新节点创建
  ListNode* tail = pHead->_prev;//找到尾节点
  tail->_next = newnode;//尾节点连接新节点
  newnode->_prev = tail;//新节点的prev与尾节点连接
  pHead->_prev = newnode;//头节点的prev指向新节点
  newnode->_next = pHead;//新节点的next指向头节点,至此,新节点成了为节点
}


尾删


尾删的实现也很简单,找到尾节点即可,再让尾节点的前一个节点与头节点连接,最后释放尾节点即可。如下:


4.gif


这里要注意的就是空表情况下是不可以继续删除的。


代码


void ListPopBack(ListNode* pHead)
{
  assert(pHead);
  assert(pHead->_next!=pHead);//空表情况下不能继续删
  //找尾
  ListNode* tail = pHead->_prev;
  //记录尾部前一个节点
  ListNode* tailprev = tail->_prev;
  tailprev->_next = pHead;
  pHead->_prev = tailprev;
  //释放尾部
  free(tail);
}


测试


//链表初始化+尾插尾删
void ListNodeTest1()
{
  //初始化
  ListNode* phead=InitListNode();
  //尾插
  ListPushBack(phead, 1);
  ListPushBack(phead, 2);
  ListPushBack(phead, 3);
  ListPushBack(phead, 4);
  ListPushBack(phead, 5);
  ListPrint(phead);//1 2 3 4 5
  //尾删
  ListPopBack(phead);
  ListPopBack(phead);
  ListPopBack(phead);
  ListPrint(phead);//1 2
  //ListPopBack(phead);
  //ListPopBack(phead);
  //ListPopBack(phead);
  //ListPrint(phead);//error
}



头插与头删

头插


头插的实现在这里也很简单,先找到有效节点的头节点(即PHead的next指向的第一个节点),然后将新节点的next指向该节点,该节点的prev指向新节点,再让PHead的next指向新节点,新节点的prev指向PHead即可。(看起来可能有些乱,但是画图就特别容易理解)


5.png


代码


//头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
  assert(pHead);
  //找到头节点
  ListNode* first = pHead->_next;
  ListNode* newnode = BuyListNode(x);//新节点创建
  //连接
  newnode->_next = first;
  first->_prev = newnode;
  pHead->_next = newnode;
  newnode->_prev = pHead;
}


头删


头删的实现与尾删异曲同工,找到有效节点的头节点,保存下一个节点,将PHead与之连接,然后再释放有效节点的头节点即可。


6.png


这里依然要注意,空表情况不能进行删除


代码


void ListPopFront(ListNode* pHead)
{
  assert(pHead);
  assert(pHead->_next != pHead);//
  //找到头节点以及头节点后面的节点
  ListNode* first = pHead->_next;
  ListNode* second = first->_next;
  //进行连接
  pHead->_next = second;
  second->_prev = pHead;
  free(first);//释放
}


测试


void ListNodeTest2()
{
  //初始化
  ListNode* phead = InitListNode();
  //头插
  ListPushFront(phead, 1);
  ListPushFront(phead, 2);
  ListPushFront(phead, 3);
  ListPushFront(phead, 4);
  ListPushFront(phead, 5);
  ListPrint(phead);//5 4 3 2 1
  //头删
  ListPopFront(phead);
  ListPopFront(phead);
  ListPopFront(phead);
  ListPrint(phead);// 2 1
  //ListPopFront(phead);
  //ListPopFront(phead);
  //ListPopFront(phead);
  //ListPrint(phead);// error
}



查找数据

这里进行查找数据,依然还是遍历整个链表即可。没啥可说的,如下:


ListNode* ListFind(ListNode* pHead, LTDataType x)
{
  assert(pHead);
  ListNode* cur = pHead->_next;
  //遍历
  while (cur != pHead)
  {
  if (cur->_data == x)
  {
    return cur;
  }
  cur = cur->_next;
  }
  return NULL;
}


在任意位置的插入与删除

pos位置插入


这里与头插的操作相比,两者其实也没啥区别。头插是找有效节点的头节点,在这里我们把pos看作该节点,把pos的prev指向的节点看作是PHead节点,这样的话,原理就与头插相同了。


7.png

// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{
  assert(pos);
  //pos前面的节点
  ListNode* posprev = pos->_prev;
  //新节点
  ListNode* newnode = BuyListNode(x);
  //连接即可
  posprev->_next = newnode;
  newnode->_prev = posprev;
  newnode->_next = pos;
  pos->_prev = newnode;
}


pos位置删除

原理也是与头删相似,只要画图即可理解。这里就不一一进行解析了,大家根据代码画图纸就行。不过需要注意的是,空表不可进行删除。


void ListErase(ListNode* pos)
{
  assert(pos);
  pos->_prev->_next = pos->_next;
  pos->_next->_prev = pos->_prev;
  free(pos);
}


链表的销毁

这里的销毁也是需要进行遍历链表,先保存下一个链表,再释放当前链表。将有效节点销毁后,再将PHead销毁。


8.gif


代码


void ListDestory(ListNode* pHead)
{
  assert(pHead);
  ListNode* cur = pHead->_next;
  while (cur != pHead)
  {
  //找到cur后面的节点
  ListNode* curnext = cur->_next;
  free(cur);//释放
  cur = curnext;
  }
  //释放pHead
  free(pHead);
}


测试


//双向链表查找、任意位置插入、删除
ListNodeTest3()
{
  ListNode* phead = InitListNode();
  ListPushFront(phead, 1);
  ListPushFront(phead, 2);
  ListPushFront(phead, 3);
  ListPushFront(phead, 4);
  ListPushFront(phead, 5);
  //查找
  ListNode* pos = ListFind(phead, 2);
  //pos->_data = 50;
  //ListPrint(phead);// 5 4 3 50 1
  // 双向链表在pos的前面进行插入
  ListInsert(pos, 0);
  ListPrint(phead);// 5 4 3 0 2 1
  //删除pos位置
  ListErase(pos);
  ListPrint(phead);// 5 4 3 0 1
  //链表销毁
  ListDestory(phead);
}


总结


该链表用起来真的很爽,能直接找到头尾节点,并且因为有头的存在,就不需要考虑是否为空表的情况下的插入,就不用改变PHead,所以就不用像之前的单链表一样,得传二级指针。真的是链表中的完美存在,不过在进行删除操作时,一定要考虑空表情况下不可进行删除。因此要加个assert进行断言。


相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
71 4
|
11天前
|
机器学习/深度学习 存储 C++
【C++数据结构——线性表】单链表的基本运算(头歌实践教学平台习题)【合集】
本内容介绍了单链表的基本运算任务,涵盖线性表的基本概念、初始化、销毁、判定是否为空表、求长度、输出、求元素值、按元素值查找、插入和删除数据元素等操作。通过C++代码示例详细解释了顺序表和链表的实现方法,并提供了测试说明、通 - **任务描述**:实现单链表的基本运算。 - **相关知识**:包括线性表的概念、初始化、销毁、判断空表、求长度、输出、求元素值、查找、插入和删除等操作。 - **测试说明**:平台会对你编写的代码进行测试,提供测试输入和预期输出。 - **通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了测试通过后的预期输出结果。 开始你的任务吧,祝你成功!
30 5
|
25天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
87 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
137 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
61 0
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
94 0
|
3月前
|
存储
[数据结构] -- 双向循环链表
[数据结构] -- 双向循环链表
31 0