• 数据结构与算法—双向链表


    目录

    一、链表的分类

    二、双向链表原理

    三、实现带头双向链表 

    1、声明链表结构体 

    2、初始化链表

    3、创建新节点

    4、打印链表

    5、头插&尾插

    头插 

    尾插

    6、头删&尾删

    头删

    尾删

    7、 查找节点

    8、 指定节点前插入

    9、 删除指定节点

    10、销毁链表

    完整版

    Lisy.h 声明

    List.c函数

    text.c测试代码


    一、链表的分类

    实际中链表的结构非常多样,以下情况组合起来就有8种链表结构:
    1. 单向或者双向
    2. 带头或者不带头(头又叫哨兵位)
    3. 循环或者非循环
    虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:
    1. 无头单向非循环链表 结构简单,一般不会单独用来存数据。实际中更多是作为 其他数据结构的子结构 ,如 哈希桶、图的邻接表 等等。另外这种结构在 笔试面试 中出现很多。
    2. 带头双向循环链表 结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多 优势 ,实现反而 简单了 ,后面我们代码实现了就知道了。

    二、双向链表原理

     我们来对比单向链表的结构体,看一下与双向链表的结构体有什么区别:(左单向,右双向)

    1. 我们可以看出双向链表比单向链表多了一个结构体指针prev,它的作用是存储前一个结点的地址。
    2. 双向链表的每个节点既可以找到前一个节点,也可以找到后一个节点,
    3. 此外头节点和尾节点比较特殊,头节点的prev指向尾节点,尾节点的next指向头节点。
    4. 如果链表只有头节点(也叫哨兵位),那么它的prev和next都指向自己。 

    三、实现带头双向链表 

    1、声明链表结构体 

    1. typedef int LTDataType;
    2. typedef struct ListNode
    3. {
    4. struct ListNode* next;
    5. struct ListNode* prev;
    6. LTDataType data;
    7. }LTNode;
    • 首先构建链表结构体,将链表结点的data数据类型设置别名LTDataType,方便后续修改。
    • 定义结构体名称的别名为LTNode。 
    • 指针next指向该节点的下一个节点,
    • 指针prev指向该节点的前一个节点。
    • LTDataType类型的data用于存值。

    2、初始化链表

    1. LTNode* LTInit()
    2. {
    3. LTNode* phead = BuyLTNode(-1);
    4. phead->prev = phead;
    5. phead->next = phead;
    6. return phead;
    7. }
    • 函数的目的是创建一个空的双向链表并返回指向链表头部的指针。
    • 通过BuyLTNode( )函数为头节点开辟空间,头节点的data值为-1。(后续讲解)
    • 初始化时,prev和next均指向头节点。
    • 返回值为指针phead。

    创建结构体指针,将初始化函数的返回值赋值给结构体指针,即为初始化完成,代码如下:

    LTNode* plist = LTInit();

      这里也可以用二级指针接收链表地址初始化链表。

    1. void LTInit(LTNode** phead)
    2. {
    3. assert(phead);
    4. * phead = BuyLTNode(-1);
    5. (*phead)->prev = *phead;
    6. (*phead)->next = *phead;
    7. (*phead)->data = 0;
    8. }

    使用该函数初始化时,使用方法也有变化:

    1. LTNode* plist;
    2. LTInit(&plist);

    3、创建新节点

    1. LTNode* BuyLTNode(LTDataType x)
    2. {
    3. LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
    4. if (newnode == NULL) {
    5. perror("malloc fail");
    6. return NULL;
    7. }
    8. newnode->data = x;
    9. newnode->prev = NULL;
    10. newnode->next = NULL;
    11. return newnode;
    12. }
    •  使用malloc为新节点newnode开辟结构体LTNode大小的空间。
    • 开辟失败perror打印错误信息,函数提前结束。
    • 为新节点的data赋值为参数x的值,prev、next置空。
    • 返回新节点指针。

    4、打印链表

    1. void LTPrint(LTNode* phead)
    2. {
    3. assert(phead);
    4. printf("guard<==>");
    5. LTNode* cur = phead->next;
    6. while (cur != phead) {
    7. if(cur!=phead->prev)
    8. printf("%d<==>", cur->data);
    9. else
    10. printf("%d\n", cur->data);
    11. cur = cur->next;
    12. }
    13. }
    • assert检查链表是否为空,为空则报错。
    • 打印字符串 "guard<==>",用于表示链表的头节点。
    • 指针cur指向头节点下一个节点,也就是第一个节点。
    • while循环的结束条件为cur指向头节点,双向链表尾部不为空,则循环遍历结束条件为尾结点的next指向头节点时。
    • 当cur不指向最后一个节点,打印节点之间的连接符号<==>。

    5、头插&尾插

    《头插&尾插 和 头删&尾删》我们都可以通过后续的指定位置前插入函数和删除函数实现, 这里进行讲解为了更好得熟悉理解双向链表

    头插 

    1. void LTPushFront(LTNode* phead, LTDataType x)
    2. {
    3. assert(phead);
    4. LTNode* newnode = BuyLTNode(x);
    5. LTNode* first = phead->next;
    6. //下面两块代码的顺序可以颠倒,因为使用指针first储存了原来链表第一个元素的地址
    7. newnode->next = first;
    8. first->prev = newnode;
    9. phead->next = newnode;
    10. newnode->prev = phead;
    11. }
    • assert检查链表是否为空,为空则报错。
    • 为新节点newnode开辟空间。
    • 指针first指向链表原第一个节点。
    • 将新节点的next指向原第一个节点,原第一个节点的prev指向新节点。
    • 头节点的next指向新节点,新节点的prev指向头节点。

     当然下面这种形式也对,只是后两个代码块不能调换顺序,具体原因大家可以画图试一下。

    1. void LTPushFront(LTNode* phead, LTDataType x)
    2. {
    3. assert(phead);
    4. LTNode* newnode = BuyLTNode(x);
    5. newnode->next = phead->next;
    6. phead->next->prev = newnode;
    7. newnode->prev = phead;
    8. phead->next = newnode;
    9. }

    尾插

    1. void LTPushBack(LTNode* phead, LTDataType x)
    2. {
    3. assert(phead);
    4. LTNode* newnode = BuyLTNode(x);
    5. LTNode* tail = phead->prev;
    6. newnode->next = phead;
    7. phead->prev = newnode;
    8. tail->next = newnode;
    9. newnode->prev = tail;
    10. }
    • assert检查链表是否为空,为空则报错。
    • 为新节点newnode开辟空间。
    • 指针tail指向尾节点,直接通过头节点phead的prev即可找到尾节点。
    • 新节点的next指向头节点,头节点的prev指向新节点。
    • 原尾节点的next指向新节点,新节点的prev指向尾节点。

    6、头删&尾删

    头删

    1. bool LTEmpty(LTNode* phead)
    2. {
    3. assert(phead);
    4. return phead->next == phead;
    5. }
    6. void LTPopFront(LTNode* phead)
    7. {
    8. assert(phead);
    9. assert(!LTEmpty(phead));
    10. LTNode* first = phead->next;
    11. LTNode* second = first->next;
    12. second->prev = phead;
    13. phead->next = second;
    14. free(first);
    15. }
    • 借助LTEmpty函数判断链表是否只有头节点,只有头节点则不进行删除。
    • 第一个assert检查链表是否为空,为空则报错。
    • 第二个assert通过LTEmpty函数返回值判断是否可以进行删除。
    • 定义两个指针first和second分别指向第一个节点和第二个节点。
    • 将第二个节点的prev指向头节点。
    • 头节点的next指向第二个节点。
    • 释放第一个节点的空间。

    尾删

    1. void LTPopBack(LTNode* phead)
    2. {
    3. assert(phead);
    4. assert(!LTEmpty(phead));
    5. LTNode* tail = phead->prev;
    6. LTNode* tailPrev = tail->prev;
    7. free(tail);
    8. tailPrev->next = phead;
    9. phead->prev = tailPrev;
    10. }
    • 第一个assert检查链表是否为空,为空则报错。
    • 第二个assert通过LTEmpty函数返回值判断是否可以进行删除。
    • 定义两个指针tail和tailPrev,分别指向尾节点和尾节点的前一个节点。
    • 释放尾节点的空间。
    • tailPrev的next指向头节点。
    • 头节点的prev指向tailPrev(新的尾节点)。

    7、 查找节点

    1. LTNode* LTFind(LTNode* phead, LTDataType x)
    2. {
    3. assert(phead);
    4. LTNode* cur = phead->next;
    5. while (cur != phead) {
    6. if (cur->data == x)
    7. return cur;
    8. cur = cur->next;
    9. }
    10. return NULL;
    11. }
    • assert检查链表是否为空,为空则报错。
    • 指针cur指向第一个节点。
    • 循环遍历链表,节点的data值为x时,返回该节点的地址。
    • 找不到则返回NULL。

    8、指定节点前插入

    1. void LTInsert(LTNode* pos, LTDataType x)
    2. {
    3. assert(pos);
    4. LTNode* prev = pos->prev;
    5. LTNode* newnode = BuyLTNode(x);
    6. prev->next = newnode;
    7. newnode->prev = prev;
    8. newnode->next = pos;
    9. pos->prev = newnode;
    10. }
    • assert检查节点是否为空,为空则报错。
    • 指针prev指向pos节点的前一个节点。
    • 为新节点newnode开辟空间,节点的data值为x。
    • 将prev的next指向新节点,新节点的prev指向prev。
    • 新节点的next指向pos,pos的prev指向新节点。

    9、删除指定节点

    1. void LTErase(LTNode* pos)//可以判断phead是否等于哨兵位
    2. {
    3. assert(pos);
    4. LTNode* posPrev = pos->prev;
    5. LTNode* posNext = pos->next;
    6. posPrev->next = posNext;
    7. posNext->prev = posPrev;
    8. free(pos);
    9. }
    • assert检查节点是否为空,为空则报错。
    • 定义两个指针posPrev和posNext分别指向 pos的前一个节点和 pos后一个节点。
    • 前一个节点posPrev的next指向posNext。
    • 后一个节点posNext的prev指向posPrev
    • 释放pos节点的空间。

    10、销毁链表

    1. void LTDestroy(LTNode** phead)
    2. {
    3. assert(phead);
    4. assert(*phead);
    5. LTNode* cur = (*phead)->next;
    6. while (cur != *phead) {
    7. LTNode* next = cur->next;
    8. free(cur);
    9. cur = next;
    10. }
    11. free(*phead);
    12. *phead = NULL;
    13. }
    • assert检查链表是否为空,为空则报错。
    • 检查链表头指针 *phead 是否为非空,为空则报错。
    • 指针cur指向第一个节点,循环内通过cur遍历每个节点。
    • next指针记录cur下一个节点,将cur的空间释放,cur更新为next,指向下一个节点。
    • 在循环结束后,链表中的所有节点都已经被销毁,最后一步是释放链表头指针 *phead 所占用的内存。

    • *phead = NULL 最后,将链表头指针设置为 NULL,以确保不再引用链表,避免悬挂指针或野指针的问题。

    完整版

    完整版代码我保留了注释,方便读者学习,以下代码的功能经测试没有问题,放心使用。 

    Lisy.h 声明

    1. #include
    2. #include
    3. #include
    4. #include
    5. typedef int LTDataType;
    6. typedef struct ListNode
    7. {
    8. struct ListNode* next;
    9. struct ListNode* prev;
    10. LTDataType data;
    11. }LTNode;
    12. //初始化链表
    13. LTNode* LTInit();
    14. //void LTInit(LTNode** phead);
    15. //打印链表
    16. void LTPrint(LTNode* phead);
    17. //头插&尾插
    18. void LTPushFront(LTNode* phead, LTDataType x);
    19. void LTPushBack(LTNode* phead, LTDataType x);
    20. //头删&尾删
    21. void LTPopFront(LTNode* phead);
    22. void LTPopBack(LTNode* phead);
    23. //查找值为x的节点,返回节点地址
    24. LTNode* LTFind(LTNode* phead, LTDataType x);
    25. //在指定节点前插入
    26. void LTInsert(LTNode* pos, LTDataType x);
    27. //删除指定节点
    28. void LTErase(LTNode* pos);
    29. //销毁链表
    30. void LTDestroy(LTNode** phead);

    List.c函数

    1. #define _CRT_SECURE_NO_WARNINGS 1
    2. #include "List.h"
    3. //创建新节点
    4. LTNode* BuyLTNode(LTDataType x)
    5. {
    6. LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
    7. if (newnode == NULL) {
    8. perror("malloc fail");
    9. return NULL;
    10. }
    11. newnode->data = x;
    12. newnode->prev = NULL;
    13. newnode->next = NULL;
    14. return newnode;
    15. }
    16. //初始化链表
    17. LTNode* LTInit()
    18. {
    19. LTNode* phead = BuyLTNode(-1);
    20. phead->prev = phead;
    21. phead->next = phead;
    22. return phead;
    23. }
    24. //void LTInit(LTNode** phead)
    25. //{
    26. // assert(phead);
    27. // * phead = BuyLTNode(-1);
    28. // (*phead)->prev = *phead;
    29. // (*phead)->next = *phead;
    30. // (*phead)->data = 0;
    31. //}
    32. //打印链表
    33. void LTPrint(LTNode* phead)
    34. {
    35. assert(phead);
    36. printf("guard<==>");
    37. LTNode* cur = phead->next;
    38. while (cur != phead) {
    39. if(cur!=phead->prev)
    40. printf("%d<==>", cur->data);
    41. else
    42. printf("%d\n", cur->data);
    43. cur = cur->next;
    44. }
    45. }
    46. //头插
    47. void LTPushFront(LTNode* phead, LTDataType x)
    48. {
    49. LTInsert(phead->next, x);
    50. }
    51. //void LTPushFront(LTNode* phead, LTDataType x)
    52. //{
    53. // assert(phead);
    54. // LTNode* newnode = BuyLTNode(x);
    55. // LTNode* first = phead->next;
    56. // //下面两块代码的顺序可以颠倒,因为使用指针first储存了原来链表第一个元素的地址
    57. // newnode->next = first;
    58. // first->prev = newnode;
    59. //
    60. // phead->next = newnode;
    61. // newnode->prev = phead;
    62. //}
    63. //void LTPushFront(LTNode* phead, LTDataType x)
    64. //{
    65. // assert(phead);
    66. // LTNode* newnode = BuyLTNode(x);
    67. //
    68. // newnode->next = phead->next;
    69. // phead->next->prev = newnode;
    70. //
    71. // newnode->prev = phead;
    72. // phead->next = newnode;
    73. //}
    74. //尾插
    75. void LTPushBack(LTNode* phead, LTDataType x)
    76. {
    77. assert(phead);
    78. LTInsert(phead, x);
    79. }
    80. //void LTPushBack(LTNode* phead, LTDataType x)
    81. //{
    82. // assert(phead);
    83. // LTNode* newnode = BuyLTNode(x);
    84. // LTNode* tail = phead->prev;
    85. //
    86. // newnode->next = phead;
    87. // phead->prev = newnode;
    88. //
    89. // tail->next = newnode;
    90. // newnode->prev = tail;
    91. //}
    92. //监测链表是否为空
    93. bool LTEmpty(LTNode* phead)
    94. {
    95. assert(phead);
    96. return phead->next == phead;
    97. }
    98. //头删
    99. void LTPopFront(LTNode* phead)
    100. {
    101. assert(phead);
    102. assert(!LTEmpty(phead));
    103. LTErase(phead->next);
    104. }
    105. //void LTPopFront(LTNode* phead)
    106. //{
    107. // assert(phead);
    108. // assert(!LTEmpty(phead));
    109. //
    110. // LTNode* first = phead->next;
    111. // LTNode* second = first->next;
    112. //
    113. // second->prev = phead;
    114. // phead->next = second;
    115. // free(first);
    116. //}
    117. //尾删
    118. void LTPopBack(LTNode* phead)
    119. {
    120. assert(phead);
    121. assert(!LTEmpty(phead));
    122. LTErase(phead->prev);
    123. }
    124. //void LTPopBack(LTNode* phead)
    125. //{
    126. // assert(phead);
    127. // assert(!LTEmpty(phead));
    128. // //LTErase(phead->prev);
    129. //
    130. // LTNode* tail = phead->prev;
    131. // LTNode* tailPrev = tail->prev;
    132. //
    133. // free(tail);
    134. // tailPrev->next = phead;
    135. // phead->prev = tailPrev;
    136. //
    137. //}
    138. //查找值为x的节点,返回节点地址
    139. LTNode* LTFind(LTNode* phead, LTDataType x)
    140. {
    141. assert(phead);
    142. LTNode* cur = phead->next;
    143. while (cur != phead) {
    144. if (cur->data == x)
    145. return cur;
    146. cur = cur->next;
    147. }
    148. return NULL;
    149. }
    150. //在指定节点前插入,与LTFind搭配使用
    151. void LTInsert(LTNode* pos, LTDataType x)
    152. {
    153. assert(pos);
    154. LTNode* prev = pos->prev;
    155. LTNode* newnode = BuyLTNode(x);
    156. prev->next = newnode;
    157. newnode->prev = prev;
    158. newnode->next = pos;
    159. pos->prev = newnode;
    160. }
    161. // 删除指定节点
    162. void LTErase(LTNode* pos)//可以判断phead是否等于哨兵位
    163. {
    164. assert(pos);
    165. LTNode* posPrev = pos->prev;
    166. LTNode* posNext = pos->next;
    167. posPrev->next = posNext;
    168. posNext->prev = posPrev;
    169. free(pos);
    170. }
    171. //销毁链表
    172. void LTDestroy(LTNode** phead)
    173. {
    174. assert(phead);
    175. assert(*phead);
    176. LTNode* cur = (*phead)->next;
    177. while (cur != *phead) {
    178. LTNode* next = cur->next;
    179. free(cur);
    180. cur = next;
    181. }
    182. free(*phead);
    183. *phead = NULL;
    184. }

    text.c测试代码

    1. #include "List.h"
    2. void test1() {
    3. /*LTNode* plist;
    4. LTInit(&plist);*/
    5. LTNode* plist = LTInit();
    6. LTPushFront(plist, 1);
    7. LTPushFront(plist, 2);
    8. LTPushFront(plist, 3);
    9. LTPushFront(plist, 4);
    10. LTPushBack(plist, 888);
    11. LTPushBack(plist, 888);
    12. LTPopFront(plist);
    13. LTPopBack(plist);
    14. LTNode* pos = LTFind(plist, 888);
    15. LTInsert(pos, 999);
    16. LTErase(pos);
    17. LTPrint(plist);
    18. LTDestroy(&plist);
    19. }
    20. int main()
    21. {
    22. test1();
    23. return 0;
    24. }

  • 相关阅读:
    【学习笔记】go协程和通道
    【算法】蓝桥杯全攻略:从语言基础到数学算法,一站式解锁竞赛技巧
    【Linux成长史】Linux权限的详细讲解
    最新版ESP32 IDF环境搭建教程:基于CLION同时安装多个版本的IDF
    【pycharm】控制台报错:终端无法加载文件\venv\Scripts\activate.ps1
    mysql数据库增量备份方案、备份计划(InsCode AI 创作助手)
    腾讯云 Web 超级播放器开发实战
    测试的流程已经测完,仍然会导致一些 BUG 流入到线上,如何规避和解决?
    基于Streamlit的YOLOv5ToX模型转换工具(适用YOLOv5训练出来的模型转化为任何格式)
    自动对话系统
  • 原文地址:https://blog.csdn.net/m0_73800602/article/details/133937579