当前位置:   article > 正文

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

双向链表

目录

一、链表的分类

二、双向链表原理

三、实现带头双向链表 

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 <stdio.h>
  2. #include <stdlib.h>
  3. #include <assert.h>
  4. #include <stdbool.h>
  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. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/419123?site
推荐阅读
相关标签
  

闽ICP备14008679号