抽象数据类型(ADT)——队列

作者:追风剑情 发布于:2020-4-22 10:34 分类:C

示例:实现一个队列

queue.h

  1. //#pragma once
  2. #ifndef _QUEUE_H_
  3. #define _QUEUE_H_
  4. #include <stdbool.h>
  5.  
  6. typedef int Item;
  7.  
  8. #define MAXQUEUE 10
  9.  
  10. typedef struct node
  11. {
  12. Item item;
  13. struct node * next;
  14. } Node;
  15.  
  16. typedef struct queue
  17. {
  18. Node * front; /* 指向队列首项的指针 */
  19. Node * rear; /* 指向队列尾项的指针 */
  20. int items; /* 队列中的项数 */
  21. } Queue;
  22.  
  23. /* 操作: 初始化队列 */
  24. /* 前提条件: pq指向一个队列 */
  25. /* 后置条件: 队列被初始化 */
  26. void InitializeQueue(Queue * pq);
  27.  
  28. /* 操作: 检查队列是否已满 */
  29. /* 前提条件: pq指向之前被初始化的队列 */
  30. /* 后置条件: 如果队列已满则返回true,否则返回false */
  31. bool QueueIsFull(const Queue * pq);
  32.  
  33. /* 操作: 检查队列是否为空 */
  34. /* 前提条件: pq指向之前被初始化的队列 */
  35. /* 后置条件: 如果队列为空则返回true,否则返回false */
  36. bool QueueIsEmpty(const Queue * pq);
  37.  
  38. /* 操作: 确定队列中的项数 */
  39. /* 前提条件: pq指向之前被初始化的队列 */
  40. /* 后置条件: 返回队列中的项数 */
  41. int QueueItemCount(const Queue * pq);
  42.  
  43. /* 操作: 在队列末尾添加项 */
  44. /* 前提条件: pq指向之前被初始化的队列 */
  45. /* item是要被添加在队列末尾的项 */
  46. /* 后置条件: 如果队列不为空,item将被添加在队列的末尾*/
  47. /* 该函数返回true;否则,队列不变,该函数返回false*/
  48. bool EnQueue(Item item, Queue * pq);
  49.  
  50. /* 操作: 从队列的开头删除项 */
  51. /* 前提条件: pq指向之前被初始化的队列 */
  52. /* 后置条件: 如果队列不为空,队列首端的item将被拷贝到*pitem中 */
  53. /* 如果该操作使得队列为空,则重置队列为空 */
  54. /* 如果队列在操作前为空,该函数返回false */
  55. bool DeQueue(Item * pitem, Queue * pq);
  56.  
  57. /* 操作: 清空队列 */
  58. /* 前提条件: pq指向之前被初始化的队列 */
  59. /* 后置条件: 队列被清空 */
  60. void EmptyTheQueue(Queue * pq);
  61.  
  62. #endif


queue.c

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include "queue.h"
  4.  
  5. /* 局部函数原型 */
  6. static void CopyToNode(Item item, Node * pn);
  7. static void CopyToItem(Node *pn, Item * pi);
  8.  
  9. /* 初始化队列 */
  10. void InitializeQueue(Queue * pq)
  11. {
  12. pq->front = pq->rear = NULL;
  13. pq->items = 0;
  14. }
  15.  
  16. /* 检查队列是否已满 */
  17. bool QueueIsFull(const Queue * pq)
  18. {
  19. return pq->items == MAXQUEUE;
  20. }
  21.  
  22. /* 检查队列是否为空 */
  23. bool QueueIsEmpty(const Queue * pq)
  24. {
  25. return pq->items == 0;
  26. }
  27.  
  28. /* 确定队列中的项数 */
  29. int QueueItemCount(const Queue * pq)
  30. {
  31. return pq->items;
  32. }
  33.  
  34. /* 在队列末尾添加项 */
  35. bool EnQueue(Item item, Queue * pq)
  36. {
  37. Node * pnew;
  38.  
  39. if (QueueIsFull(pq))
  40. return false;
  41. pnew = (Node *)malloc(sizeof(Node));
  42. if (pnew == NULL)
  43. {
  44. fprintf(stderr, "Unable to allocate memory!\n");
  45. exit(1);
  46. }
  47. CopyToNode(item, pnew);
  48. pnew->next = NULL;
  49. if (QueueIsEmpty(pq))
  50. pq->front = pnew; /* 项位于队列首端 */
  51. else
  52. pq->rear->next = pnew; /* 链接到队列尾端 */
  53. pq->rear = pnew; /* 记录队列尾端的位置 */
  54. pq->items++; /* 队列项数加1 */
  55.  
  56. return true;
  57. }
  58.  
  59. /* 从队列的开头删除项 */
  60. bool DeQueue(Item * pitem, Queue * pq)
  61. {
  62. Node * pt;
  63.  
  64. if (QueueIsEmpty(pq))
  65. return false;
  66. CopyToItem(pq->front, pitem);
  67. pt = pq->front;
  68. pq->front = pq->front->next;
  69. free(pt);
  70. pq->items--;
  71. if (pq->items == 0)
  72. pq->rear = NULL;
  73.  
  74. return true;
  75. }
  76.  
  77. /* 清空队列 */
  78. void EmptyTheQueue(Queue * pq)
  79. {
  80. Item dummy;
  81. while (!QueueIsEmpty(pq))
  82. DeQueue(&dummy, pq);
  83. }
  84.  
  85. /* 局部函数 */
  86.  
  87. static void CopyToNode(Item item, Node * pn)
  88. {
  89. pn->item = item;
  90. }
  91.  
  92. static void CopyToItem(Node *pn, Item * pi)
  93. {
  94. *pi = pn->item;
  95. }


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 "queue.h"
  14.  
  15. int main(int argc, char* argv[])
  16. {
  17. Queue line;
  18. Item temp;
  19. char ch;
  20.  
  21. InitializeQueue(&line);
  22. puts("Testing the Queue interface. Type a to add a value, ");
  23. puts("type d to delete a value, and type q to quit.");
  24. while ((ch = getchar()) != 'q')
  25. {
  26. if (ch != 'a' && ch != 'd') /* 忽略其他输出 */
  27. continue;
  28. if (ch == 'a')
  29. {
  30. printf("Integer to add: ");
  31. scanf("%d", &temp);
  32. if (!QueueIsFull(&line))
  33. {
  34. printf("Putting %d into queue\n", temp);
  35. EnQueue(temp, &line);
  36. }
  37. else
  38. puts("Queue is full!");
  39. }
  40. else
  41. {
  42. if (QueueIsEmpty(&line))
  43. puts("Nothing to delete!");
  44. else
  45. {
  46. DeQueue(&temp, &line);
  47. printf("Removing %d from queue\n", temp);
  48. }
  49. }
  50. printf("%d items in queue\n", QueueItemCount(&line));
  51. puts("Type a to add, d to delete, q to quit:");
  52. }
  53. EmptyTheQueue(&line);
  54. puts("Bye!");
  55.  
  56. system("pause");
  57. return 0;
  58. }


运行测试
11111.png

标签: C语言

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号