当前位置:   article > 正文

数据结构之双向链表_双向链表数据结构

双向链表数据结构

双向链表与单向链表较为类似,单向链表有一个指针域,用来指向后继结点,而双向链表有两个指针域,分别用来指向前驱结点和后继结点。玩双向链表时一定要从单向链表的思维中跳出来,否则在操作双向链表时就会出现各种问题。在玩双向链表时,一定要动手画图或者在脑子里画图,考虑好每个操作细节。 

 一、双向链表的基本操作

双向链表的数据结构:

  1. typedef struct node//定义双向链表的数据结构
  2. {
  3. int data;
  4. struct node * prior;
  5. struct node * next;
  6. }Node;

这里以整型作为结点数据域的数据类型

1. 初始化链表

        Node * initialize();

       创建头结点,初始化头结点的各个成员变量,返回头结点的指针

 2.头插法添加结点

        void insertFirst(Node * head,int data);

        方法形参上需传入头结点指针和新结点的数据,方法内会创建新结点,之后操作新结点、头结点、首结点,将新结点插入首结点的位置

3.尾插法添加结点

        void append(Node * head,int data);

        方法形参上需传入头结点指针和新结点的数据,方法内会创建新结点,之后操作头结点、末结点、新结点,将新结点插入到末结点的后面

4.删除结点

        int delete(Node * head,int data);

         方法形参上需传入头结点指针和结点的数据,方法内会从首结点向末结点遍历,查找指定数据的结点,若查找成功,则删除该结点,返回1,若查找无果,返回0

5.反转链表

        void reverse(Node * head);

          方法形参上需传入头结点指针,方法内会将链表的首尾顺序进行颠倒,原先的首结点作尾结点,原先的尾结点作首结点

6.正向遍历链表

        void traverse(Node * head);

          方法形参上需传入头结点指针,方法内输出链表的所有结点数据,不包括头结点

7.反向遍历链表

        void traverseReversely(Node * head);

        方法形参上需传入头结点指针,方法内从末结点到首结点进行数据输出,不输出头结点

二、代码实现

  1. /*
  2. 双向链表
  3. 初始化链表
  4. 增加结点(头插法、尾插法)
  5. 删除结点
  6. 反转链表
  7. 遍历链表(正向遍历,反向遍历)
  8. */
  9. #include<stdio.h>
  10. #include<stdlib.h>
  11. typedef struct node//定义双向链表的数据结构
  12. {
  13. int data;
  14. struct node * prior;
  15. struct node * next;
  16. }Node;
  17. Node * initialize();//初始化链表
  18. void insertFirst(Node * head,int data);//头插法添加结点
  19. void append(Node * head,int data);//尾插法添加结点
  20. int delete(Node * head,int data);//删除结点
  21. void reverse(Node * head);//反转链表
  22. void traverse(Node * head);//正向遍历链表
  23. void traverseReversely(Node * head);//反向遍历链表
  24. int main(int argc,char * argv[])
  25. {
  26. puts("------------------初始化链表------------------");
  27. Node * head=initialize();
  28. printf("head->prior=%p\n",head->prior);
  29. printf("head->next=%p\n",head->next);
  30. printf("head->data=%i\n",head->data);
  31. puts("------------------头插法插入30,50,80------------------");
  32. insertFirst(head,30);
  33. insertFirst(head,50);
  34. insertFirst(head,80);
  35. puts("------------------正向遍历链表------------------");
  36. traverse(head);
  37. puts("------------------反向遍历链表------------------");
  38. traverseReversely(head);
  39. puts("------------------尾插法插入18,19,20------------------");
  40. append(head,18);
  41. append(head,19);
  42. append(head,20);
  43. puts("------------------正向遍历链表------------------");
  44. traverse(head);
  45. puts("------------------反向遍历链表------------------");
  46. traverseReversely(head);
  47. puts("------------------删除数据为20的结点------------------");
  48. puts(delete(head,20)?"删除成功":"删除失败");
  49. traverse(head);
  50. puts("------------------删除数据为80的结点------------------");
  51. puts(delete(head,80)?"删除成功":"删除失败");
  52. traverse(head);
  53. puts("------------------删除数据为30的结点------------------");
  54. puts(delete(head,30)?"删除成功":"删除失败");
  55. traverse(head);
  56. puts("------------------尾插法插入90,150------------------");
  57. append(head,90);
  58. append(head,150);
  59. traverse(head);
  60. puts("------------------反转链表------------------");
  61. reverse(head);
  62. traverse(head);
  63. puts("------------------再反转链表------------------");
  64. reverse(head);
  65. traverse(head);
  66. return 0;
  67. }
  68. Node * initialize()//初始化链表
  69. {
  70. Node * head=(Node *)malloc(sizeof(Node));
  71. head->data=0;//头结点的数据域用来存储链表中结点的数量,不包括头结点
  72. head->prior=NULL;//头结点的前指针域置为空,在整个程序的运行过程中不作改变
  73. head->next=NULL;//头结点的后指针域暂时置为空,在整个程序的运行过程中会产生变化
  74. return head;
  75. }
  76. void insertFirst(Node * head,int data)//头插法添加结点
  77. {
  78. Node * newborn=(Node *)malloc(sizeof(Node));//创建新结点
  79. newborn->data=data;//为新结点的数据域赋值
  80. /*
  81. 采用头插法为双向链表添加结点时,
  82. 注意:链表中只有一个头结点时,有三个指针变量需要赋值,
  83. 分别是新结点前指针域、后指针域,头结点的后指针域
  84. 链表中除了头结点外,还存在其它结点,有四个指针变量需要赋值
  85. 分别是新结点前指针域、后指针域,原先的首结点的前指针域,头结点的后指针域
  86. */
  87. newborn->prior=head;
  88. newborn->next=head->next;
  89. if(head->next!=NULL)//链表中不是只有头结点,头结点有后继结点,该后继结点就是首结点
  90. {
  91. head->next->prior=newborn;//将首结点的前指针域指向新结点
  92. }
  93. head->next=newborn;//头结点的后指针域指向新结点
  94. head->data++;//头结点的数据域自增
  95. }
  96. void append(Node * head,int data)//尾插法添加结点
  97. {
  98. Node * begin=head;//定义指针变量begin,初始值为头结点的指针,将其作为循环变量
  99. for(;begin->next!=NULL;begin=begin->next);//遍历至末结点
  100. Node * newborn=(Node *)malloc(sizeof(Node));//创建新结点
  101. newborn->data=data;//为新结点的数据域赋值
  102. newborn->prior=begin;//新结点的前指针域指向末结点
  103. newborn->next=begin->next;//把末结点后指针域的值赋给新结点的后指针域
  104. begin->next=newborn; //末结点的后指针域指向新结点
  105. head->data++;//链表中结点数量自增
  106. }
  107. int delete(Node * head,int data)//删除结点
  108. {
  109. for(Node * begin=head->next;begin;begin=begin->next)//从首结点向尾结点进行遍历
  110. {
  111. if(begin->data==data)//若查找到指定数据的结点,先问问该结点是否为末结点
  112. {
  113. if(begin->next!=NULL)//若非末结点
  114. {
  115. begin->prior->next=begin->next;
  116. begin->next->prior=begin->prior;
  117. }else//若为末结点
  118. {
  119. begin->prior->next=begin->next;
  120. }
  121. free(begin);//释放要删除的结点空间
  122. head->data--;//链表中的结点数量自减
  123. return 1;
  124. }
  125. }
  126. return 0;
  127. }
  128. void reverse(Node * head)//反转链表
  129. {
  130. //反转链表使用头插法完成
  131. Node * temporary;//临时变量,作一个中间存储
  132. Node * begin=head->next;//定义指针变量begin,存储首结点的指针
  133. head->next=NULL;//头结点的后指针域置为空
  134. while(begin)//从首结点向末结点进行遍历
  135. {
  136. temporary=begin->next;//将当前结点的后继结点指针存储到指针变量temporary中,
  137. //当前结点的两个指针域重新赋值后,就无法确定原先的后继结点,这里先把其后续结点指针存储起来
  138. begin->prior=head;//当前结点的前指针域指向头结点
  139. begin->next=head->next;//当前结点的后指针域指向头结点后指针域的指向
  140. if(head->next!=NULL)//若头结点后指针域不为空,则说明链表中不是只有头结点
  141. {
  142. head->next->prior=begin;//头结点的后继结点前指针域指向当前结点
  143. }
  144. head->next=begin;//头结点的后指针域指向当前结点
  145. begin=temporary;//指针变量begin从当前结点转到下一个结点
  146. }
  147. }
  148. void traverse(Node * head)//正向遍历链表
  149. {
  150. //头结点的数据域不进行展示,这里直接用head->next表示其它结点
  151. //用前驱的后指针域表示后继结点
  152. if(head->next->next==NULL)//到达末结点
  153. {
  154. printf("%d\n",head->next->data);
  155. return;
  156. }
  157. printf("%d , ",head->next->data);
  158. traverse(head->next);
  159. }
  160. void traverseReversely(Node * head)//反向遍历链表
  161. {
  162. Node * begin=head->next;//将首结点的指针赋给指针变量begin
  163. while(begin->next!=NULL)//将指针变量begin作为循环变量进行遍历,循环条件是begin未到达末结点
  164. {
  165. begin=begin->next;//指针变量begin向末结点遍历
  166. }
  167. while(begin!=head)
  168. {
  169. if(begin->prior==head)//到达首结点
  170. {
  171. printf("%d\n",begin->data);
  172. }else
  173. {
  174. printf("%d , ",begin->data);
  175. }
  176. begin=begin->prior;
  177. }
  178. }

三、运行展示

 四、总结

        玩双向链表,画图第一步,代码第二步,运行第三步。细心、耐心、恒心即可把玩双链表。

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

闽ICP备14008679号