抽象数据类型(ADT)——链表

作者:追风剑情 发布于:2020-4-21 11:23 分类:C

示例:通过链表储存用户输入的影片信息

list.h

  1. //#pragma once
  2. //防止多次包含头文件
  3. #ifndef LIST_H_
  4. #define LIST_H_
  5. #include <stdbool.h> /* C99特性 */
  6.  
  7. /* 特定程序的声明 */
  8.  
  9. #define TSIZE 45 /* 储存电影名的数组大小 */
  10. struct film
  11. {
  12. char title[TSIZE];
  13. int rating;
  14. };
  15.  
  16. /* 一般类型定义 */
  17. typedef struct film Item;
  18.  
  19. typedef struct node
  20. {
  21. Item item;
  22. struct node * next;
  23. } Node;
  24.  
  25. typedef Node * List;
  26.  
  27. /* List的另一种定义 */
  28. /*
  29. typedef struct list
  30. {
  31. Node * head; //指向链表头的指针
  32. Node * end; //指向链表尾的指针
  33. int size; //链表中的项数
  34. } List;
  35. */
  36.  
  37. /* 函数原型 */
  38.  
  39. /* 操作: 初始化一个链表 */
  40. /* 前提条件: plist指向一个链表 */
  41. /* 后置条件: 链表初始化为空 */
  42. void InitializeList(List * plist);
  43.  
  44. /* 操作: 确定链表是否为空,plist指向一个已初始化的链表 */
  45. /* 后置条件: 如果链表为空,该函数返回true;否则返回false */
  46. bool ListIsEmpty(const List * plist);
  47.  
  48. /* 操作: 确定链表是否已满,plist指向一个已初始化的链表 */
  49. /* 后置条件: 如果链表已满,该函数返回true;否则返回false */
  50. bool ListIsFull(const List * plist);
  51.  
  52. /* 操作: 确定链表中的项数,plist指向一个已初始化的链表 */
  53. /* 后置条件: 该函数返回链表中的基数 */
  54. unsigned int ListItemCount(const List * plist);
  55.  
  56. /* 操作: 在链表的末尾添加项 */
  57. /* 前提条件: item是一个待添加至链表的项,plist指向一个已初始化的链表 */
  58. /* 后置条件: 如果可以,该函数在链表末尾添加一个项,且返回true,否则返回false */
  59. bool AddItem(Item item, List * plist);
  60.  
  61. /* 操作: 把函数作用于链表中的每一项 */
  62. /* plist指向一个已初始化的链表 */
  63. /* pfun指向一个函数,该函数接受一个Item类型的参数,且无返回值 */
  64. /* 后置条件: pfun指向的函数作用于链表中的每一项一次 */
  65. void Traverse(const List * plist, void(*pfun)(Item item));
  66.  
  67. /* 操作: 释放已分配的内存(如果有的话) */
  68. /* plist指向一个已初始化的链表 */
  69. void EmptyTheList(List * plist);
  70.  
  71. #endif


list.c

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include "list.h"
  4.  
  5. /* 局部函数原型 */
  6. static void CopyToNode(Item item, Node * pnode);
  7.  
  8. void InitializeList(List * plist)
  9. {
  10. *plist = NULL;
  11. }
  12.  
  13. /* 如果链表为空,返回true */
  14. bool ListIsEmpty(const List * plist)
  15. {
  16. if (*plist == NULL)
  17. return true;
  18. return false;
  19. }
  20.  
  21. /* 如果链表已满, 返回true */
  22. bool ListIsFull(const List * plist)
  23. {
  24. Node * pt;
  25. bool full;
  26.  
  27. pt = (Node *)malloc(sizeof(Node));
  28. if (pt == NULL) //分配内存失败
  29. full = true;
  30. else
  31. full = false;
  32. free(pt);
  33. return full;
  34. }
  35.  
  36. /* 返回节点的数量 */
  37. unsigned int ListItemCount(const List * plist)
  38. {
  39. unsigned int count = 0;
  40. Node * pnode = *plist; /* 设置链表的开始 */
  41.  
  42. while (pnode != NULL)
  43. {
  44. ++count;
  45. pnode = pnode->next; /* 设置下一个节点 */
  46. }
  47. return count;
  48. }
  49.  
  50. /* 创建储存项的节点,并将其添加至由plist指向的链表末尾(较慢的实现) */
  51. bool AddItem(Item item, List * plist)
  52. {
  53. Node * pnew;
  54. Node * scan = *plist;
  55. pnew = (Node *)malloc(sizeof(Node));
  56. if (pnew == NULL) //分配内存失败
  57. return false;
  58. CopyToNode(item, pnew);
  59. pnew->next = NULL;
  60. if (scan == NULL) //如果链表为空
  61. *plist = pnew;
  62. else
  63. {
  64. while (scan->next != NULL)
  65. scan = scan->next; //找到链表的末尾
  66. scan->next = pnew; //把pnew添加到链表的末尾
  67. }
  68. return true;
  69. }
  70.  
  71. /* 访问每个节点并执行pfun指向的函数 */
  72. void Traverse(const List * plist, void(*pfun)(Item item))
  73. {
  74. Node * pnode = *plist; /* 设置链表的开始 */
  75. while (pnode != NULL)
  76. {
  77. (*pfun)(pnode->item); /* 把函数应用于链表中的项 */
  78. pnode = pnode->next;
  79. }
  80. }
  81.  
  82. /* 释放由malloc()分配的内存 */
  83. /* 释放链表指针为NULL */
  84. void EmptyTheList(List * plist)
  85. {
  86. Node * psave;
  87. while (*plist != NULL)
  88. {
  89. psave = (*plist)->next;
  90. free(*plist);
  91. *plist = psave;
  92. }
  93. }
  94.  
  95. /* 局部函数 */
  96. /* 把一个项拷贝到节点中 */
  97. static void CopyToNode(Item item, Node * pnode)
  98. {
  99. pnode->item = item;
  100. }


Main.c

  1. #define _CRT_SECURE_NO_WARNINGS
  2. //可以用#ifndef指令,防止多次包含一个文件
  3. #include <stdio.h>
  4. //提供malloc()原型
  5. #include <stdlib.h>
  6. //提供strcpy()原型
  7. #include <string.h>
  8. //提供CHAR_BIT的定义,CHAR_BIT表示每字节的位数
  9. #include <limits.h>
  10. //C99定义了bool、true、false
  11. //如果编译器不支持bool,可以用枚举替代enum bool {false, true};
  12. #include <stdbool.h>
  13. #include "list.h"
  14.  
  15. void showmovies(Item item);
  16. char* s_gets(char* st, int n);
  17.  
  18. int main(int argc, char* argv[])
  19. {
  20. List movies;
  21. Item temp;
  22.  
  23. /* 初始化 */
  24. InitializeList(&movies);
  25. if (ListIsFull(&movies))
  26. {
  27. fprintf(stderr, "No memory available! Bye!\n");
  28. exit(1);
  29. }
  30.  
  31. /* 获取用户输入并储存 */
  32. puts("Enter first movie title:");
  33. while (s_gets(temp.title, TSIZE) != NULL &&
  34. temp.title[0] != '\0')
  35. {
  36. puts("Enter your rating<0-10>:");
  37. scanf("%d", &temp.rating);
  38. while (getchar() != '\n')
  39. continue;
  40. if (AddItem(temp, &movies) == false)
  41. {
  42. fprintf(stderr, "Problem allocating memory\n");
  43. break;
  44. }
  45. if (ListIsFull(&movies))
  46. {
  47. puts("The list is now full.");
  48. break;
  49. }
  50. puts("Enter next movie title (empty line to stop):");
  51. }
  52.  
  53. /* 显示 */
  54. if (ListIsEmpty(&movies))
  55. printf("No data entered.");
  56. else
  57. {
  58. printf("Here is the movie list:\n");
  59. Traverse(&movies, showmovies);
  60. }
  61. printf("You entered %d movies.\n", ListItemCount(&movies));
  62. /* 清理 */
  63. EmptyTheList(&movies);
  64. printf("Bye!\n");
  65.  
  66. system("pause");
  67. return 0;
  68. }
  69.  
  70. void showmovies(Item item)
  71. {
  72. printf("Movie: %s Rating: %d\n", item.title, item.rating);
  73. }
  74.  
  75. // 自己实现读取函数
  76. char* s_gets(char* st, int n)
  77. {
  78. char* ret_val;
  79. int i = 0;
  80. ret_val = fgets(st, n, stdin);
  81. if (ret_val) //即,ret_val != NULL
  82. {
  83. while (st[i] != '\n' && st[i] != '\0')
  84. i++;
  85. if (st[i] == '\n')
  86. st[i] = '\0';
  87. else
  88. while (getchar() != '\n')
  89. continue;
  90. }
  91. return ret_val;
  92. }


运行测试
1111.png

标签: C语言

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号