当前位置:   article > 正文

C 语言动态链表

C 语言动态链表

线性结构->顺序存储->动态链表

一、理论部分

从起源中理解事物,就是从本质上理解事物。 -杜勒鲁奇

动态链表是通过结点(Node)的集合来非连续地存储数据,结点之间通过指针相互连接。

动态链表本身就是一种动态分配内存的数据结构。每个结点都包含数据部分和指向下一个节点的指针。这种结构允许在运行时动态地添加、删除或修改结点,而不需要像数组那样担心容量问题。

1.1、画图的重要性

直观理解数据结构:画图可以帮助我们直观地理解链表的结构,包括每个节点的位置、存储的数据以及节点之间的连接关系。这种直观的理解有助于我们更好地编写、调试和维护链表相关的代码。

辅助算法设计:在设计链表操作的算法时,如插入、删除、查找等,画图可以帮助我们清晰地展示算法的执行步骤和过程。通过画图,我们可以模拟算法的执行,预测结果,并发现潜在的错误或问题。这种可视化的方法对于复杂算法的设计和实现尤为重要。

提高调试效率:在链表操作中,很容易出现指针错误,如野指针、空指针解引用、内存泄漏等。通过画图,我们可以跟踪链表的状态变化,检查指针的指向是否正确,从而快速定位和解决这些问题。此外,画图还可以帮助我们理解程序的执行流程,找出逻辑上的错误或不合理之处。

促进团队合作与交流:在团队开发项目中,链表等数据结构的使用和操作往往是必不可少的。通过画图,我们可以将链表的结构和算法的执行过程清晰地展示给团队成员,促进彼此之间的理解和交流。这种可视化的沟通方式有助于提高团队的工作效率,减少误解和错误。

辅助教学与学习:对于初学者来说,链表等复杂数据结构可能难以理解和掌握。通过画图的方式,教师可以更加直观地讲解链表的结构和操作方法,帮助学生建立正确的认知模型。而学生也可以通过画图来巩固所学知识,加深对链表等数据结构的理解。

二、画图+代码分析

需求:动态链表实现对整型数据的增删改查。

创建头文件,定义如下:

  1. typedef int data_type;
  2. struct node{
  3. data_type data;
  4. struct node *next;
  5. };
  6. typedef struct node listnode;
  7. typedef enum{
  8. OK = 0,
  9. HEAD_NULL,
  10. LIST_MEMORY_ERROR,
  11. LIST_EMPTY,
  12. NO_SUCH_ELEMENT,
  13. }ERRORNUM;

2.1、创建头结点

如图所示:创建头结点

步骤:
①:在堆上申请struct node大小的一片地址空间;
②:将数据域清零,下一个节点指针域指向NULL;
③:返回头结点。

  1. listnode *create_head(void)
  2. {
  3. listnode *pnode = (listnode *)malloc(sizeof(listnode));
  4. if(pnode == NULL){
  5. perror("create_head_fail");
  6. return NULL;
  7. }
  8. pnode->data = 0;
  9. pnode->next = NULL;
  10. return pnode;
  11. }

2.2、插入结点

插入方式:头插

①:先创建一个结点指针变量,malloc申请空间;
②:将头结点后的结点(有效结点)赋值给当前要插入的结点的next域,目的在于保护尾部结点
        代码:pnew->next = head->next;
③:将当前插入的新节点赋值给head->next域    代码:head->next = pnew;
④:将插入的数据通过形参传值的方式,赋值给pnew->data域。

  1. int insert_head(listnode *head, data_type data)
  2. {
  3. if(head == NULL){
  4. return HEAD_NULL;
  5. }
  6. listnode *pnew = (listnode *)malloc(sizeof(listnode));
  7. if(pnew == NULL){
  8. return LIST_MEMORY_ERROR;
  9. }
  10. pnew->next = head->next;
  11. head->next = pnew;
  12. pnew->data = data;
  13. return OK;
  14. }

2.3、遍历打印结点元素

①:将头结点通过形参传进来;

②:入参检测:<1.>判断头结点是否为空;<2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。如果只有头结点,没有有效数据节点,则遍历无意义;

③:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。因为:

        1、简化遍历过程:使用current指针可以在遍历链表时不必每次都从头节点开始。通过移动current指针,可以逐个访问链表中的每个节点,直到到达链表的末尾(即current为NULL)。

        2、保持链表结构的完整性:直接操作头指针(如head)来遍历链表可能会不小心改变链表的头部,尤其是在某些复杂的操作中(如删除头节点)。使用current这样的临时指针可以避免这种风险,因为它只是指向链表中的一个节点,而不是链表的头部。

        3、提高代码的可读性和可维护性:使用明确的变量名(current)来表示当前正在处理的节点,可以使代码更加清晰易懂。这对于维护代码和与他人协作非常有帮助。

④:循环打印结点元素。

  1. int list_show(listnode *head)
  2. {
  3. if(head == NULL || head->next == NULL){
  4. return LIST_EMPTY;
  5. }
  6. listnode *current = head->next;
  7. // 注意,遍历链表时,判断条件式永远是当前的结点
  8. // 不是当前结点的下一个结点,否则在打印时会丢失最后一个结点
  9. while(current != NULL){
  10. printf("%d ", current->data);
  11. current = current->next;
  12. }
  13. putchar(10);
  14. return OK;
  15. }

2.4、删除结点

①:入参检测:<1.>判断头结点是否为空;<2.>判断头结点后的首节点是否为空(因为其存的是有效数据);

②:定义listnode结构体指针变量pnode,将头结点赋值给它。因为:保持链表结构的完整性:直接操作头指针(如head)来遍历链表可能会不小心改变链表的头部;

③:定义listnode结构体指针变量temp。因为:

        1、保护链表结构:要删除链表中的一个节点时,需要先找到该节点的前一个节点(假设不是删除头节点)。然后,需要将前一个节点的next指针绕过被删除的节点,直接指向被删除节点的下一个节点。如果直接操作被删除节点的next指针来修改链表,那么可能会丢失对被删除节点下一个节点的引用,从而可能导致内存泄漏或无法访问链表的剩余部分。

        2、安全地释放内存:在C语言中,动态分配的内存(如使用malloc或calloc等函数分配的内存)在使用完毕后应该被释放,以避免内存泄漏。通过定义一个temp指针来指向被删除的节点,可以在被删除节点从链表中移除之前安全地获取其地址,并使用free函数释放该节点的内存。

        3、避免复杂的条件语句:在某些情况下,特别是在处理头节点或尾节点删除时,直接修改链表可能会引入复杂的条件语句来区分不同的情况。通过使用temp指针,可以编写更清晰、更易于维护的代码来处理这些情况。

  1. int list_delete(listnode *head, data_type data)
  2. {
  3. if(head == NULL || head->next ==NULL){
  4. return LIST_EMPTY;
  5. }
  6. listnode *phead = head;
  7. listnode *temp;
  8. while(phead != NULL){
  9. if(phead->next->data == data){
  10. temp = phead->next;
  11. phead->next = phead->next->next;
  12. free(temp);
  13. return OK;
  14. }
  15. phead = phead->next;
  16. }
  17. return NO_SUCH_ELEMENT;
  18. }

2.5、修改节点元素

①:入参检测:
        <1.>判断头结点是否为空;
        <2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。
如果只有头结点,没有有效数据节点,则修改无意义;
②:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。原因这里不再赘述;
③遍历,将旧值赋值给新值。

  1. int list_modify(listnode *head, data_type old_data, data_type new_data)
  2. {
  3. if(head == NULL || head->next == NULL){
  4. return LIST_EMPTY;
  5. }
  6. listnode *current = head->next;
  7. while(current != NULL){
  8. if(current->data == old_data){
  9. current->data = new_data;
  10. return OK;
  11. }
  12. current = current->next;
  13. }
  14. return NO_SUCH_ELEMENT;
  15. }

2.6、查询结点元素

①:入参检测:
        <1.>判断头结点是否为空;
        <2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。
如果只有头结点,没有有效数据节点,则修改无意义。
②:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。
原因这里不再赘述。
③遍历,将找到的结点值打印

  1. int list_search(listnode *head, data_type data)
  2. {
  3. if(head == NULL || head->next == NULL){
  4. return LIST_EMPTY;
  5. }
  6. int count = 0;
  7. listnode *current = head->next;
  8. while(current != NULL){
  9. if(current->data == data){
  10. printf("The element->%d is located at the %d node\n", data, count+1);
  11. return OK;
  12. }
  13. count++;
  14. current = current->next;
  15. }
  16. return NO_SUCH_ELEMENT;
  17. }

三、源码

3.1、目录结构

主目录Makefile

  1. ALL:
  2. make -C ./src/
  3. make -C ./obj/
  4. .PHONY: clean
  5. clean:
  6. rm obj/*.o
  7. rm bin/*

3.2、include目录

test.h

  1. #ifndef _TEST_H_
  2. #define _TEST_H_
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. typedef int data_type;
  6. struct node{
  7. data_type data;
  8. struct node *next;
  9. };
  10. typedef struct node listnode;
  11. typedef enum{
  12. OK = 0,
  13. HEAD_NULL,
  14. LIST_MEMORY_ERROR,
  15. LIST_EMPTY,
  16. NO_SUCH_ELEMENT,
  17. }ERRORNUM;
  18. listnode *create_head(void);
  19. int insert_head(listnode *head, data_type data);
  20. int list_show(listnode *head);
  21. int list_delete(listnode *head, data_type data);
  22. int list_modify(listnode *head, data_type old_data, data_type new_data);
  23. int list_search(listnode *head, data_type data);
  24. #endif

3.3、obj目录

Makefile

  1. ALL:
  2. gcc *.o -o ../bin/app

3.4、src目录

crud.c

  1. #include "../include/test.h"
  2. listnode *create_head(void)
  3. {
  4. listnode *pnode = (listnode *)malloc(sizeof(listnode));
  5. if(pnode == NULL){
  6. perror("create_head_fail");
  7. return NULL;
  8. }
  9. pnode->data = 0;
  10. pnode->next = NULL;
  11. return pnode;
  12. }
  13. int insert_head(listnode *head, data_type data)
  14. {
  15. if(head == NULL){
  16. return HEAD_NULL;
  17. }
  18. listnode *pnew = (listnode *)malloc(sizeof(listnode));
  19. if(pnew == NULL){
  20. return LIST_MEMORY_ERROR;
  21. }
  22. pnew->next = head->next;
  23. head->next = pnew;
  24. pnew->data = data;
  25. return OK;
  26. }
  27. int list_show(listnode *head)
  28. {
  29. if(head == NULL || head->next == NULL){
  30. return LIST_EMPTY;
  31. }
  32. listnode *current = head->next;
  33. // 注意,遍历链表时,判断条件式永远是当前的结点
  34. // 不是当前结点的下一个结点,否则在打印时会丢失最后一个结点
  35. while(current != NULL){
  36. printf("%d ", current->data);
  37. current = current->next;
  38. }
  39. putchar(10);
  40. return OK;
  41. }
  42. int list_delete(listnode *head, data_type data)
  43. {
  44. if(head == NULL || head->next ==NULL){
  45. return LIST_EMPTY;
  46. }
  47. listnode *phead = head;
  48. listnode *temp;
  49. while(phead != NULL){
  50. if(phead->next->data == data){
  51. temp = phead->next;
  52. phead->next = phead->next->next;
  53. free(temp);
  54. return OK;
  55. }
  56. phead = phead->next;
  57. }
  58. return NO_SUCH_ELEMENT;
  59. }
  60. int list_modify(listnode *head, data_type old_data, data_type new_data)
  61. {
  62. if(head == NULL || head->next == NULL){
  63. return LIST_EMPTY;
  64. }
  65. listnode *current = head->next;
  66. while(current != NULL){
  67. if(current->data == old_data){
  68. current->data = new_data;
  69. return OK;
  70. }
  71. current = current->next;
  72. }
  73. return NO_SUCH_ELEMENT;
  74. }
  75. int list_search(listnode *head, data_type data)
  76. {
  77. if(head == NULL || head->next == NULL){
  78. return LIST_EMPTY;
  79. }
  80. int count = 0;
  81. listnode *current = head->next;
  82. while(current != NULL){
  83. if(current->data == data){
  84. printf("The element->%d is located at the %d node\n", data, count+1);
  85. return OK;
  86. }
  87. count++;
  88. current = current->next;
  89. }
  90. return NO_SUCH_ELEMENT;
  91. }

main.c

  1. #include "../include/test.h"
  2. int main()
  3. {
  4. int ret = 0;
  5. listnode *head = NULL;
  6. head = create_head();
  7. if(head == NULL){
  8. return -1;
  9. }
  10. head = create_head();
  11. ret = insert_head(head, 15);
  12. ret = insert_head(head, 10);
  13. ret = insert_head(head, 5);
  14. ret = list_show(head);
  15. //ret = list_delete(head, 5);
  16. //et = list_show(head);
  17. //ret = list_modify(head, 30, 8);
  18. //ret = list_show(head);
  19. ret = list_search(head, 20);
  20. switch(ret){
  21. /* 通常不需要打印"成功"信息,但可以根据需要添加
  22. case SHOW_OK:
  23. printf("SHOW_OK\n");
  24. break;
  25. */
  26. case HEAD_NULL:
  27. printf("HEAD_NULL\n");
  28. break;
  29. case LIST_MEMORY_ERROR:
  30. printf("LIST_MEMORY_ERROR\n");
  31. break;
  32. case LIST_EMPTY:
  33. printf("LIST_EMPTY\n");
  34. break;
  35. case NO_SUCH_ELEMENT:
  36. printf("NO_SUCH_ELEMENT\n");
  37. break;
  38. }
  39. return 0;
  40. }

Makefile

  1. ALL:../obj/main.o ../obj/crud.o
  2. ../obj/main.o:main.c
  3. gcc -c $< -o $@
  4. ../obj/crud.o:crud.c
  5. gcc -c $< -o $@

四、ReadMe

比较简单的顺序存储,动态链表
    返回值是通过枚举实现,删改查的函数在main函数里有个小bug,比如删除的元素不存在,错误信息会正常返回,但要是调用显示函数的话,会把错误码覆盖掉,
这就导致看不到错误原因。改查同样如此,如果此程序改为与用于交互的话,这个bug可以很好解决掉。
    程序当中有很多变量名需要优化,不是很见名知意。

五、源码下载

链接:https://pan.baidu.com/s/1ifx7ZCO7mt_HTQ76TUS33w 
提取码:ckr8

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

闽ICP备14008679号