示例:用队列模拟排队

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

示例:排队咨询

queue.h

  1. //#pragma once
  2. #ifndef _QUEUE_H_
  3. #define _QUEUE_H_
  4. #include <stdbool.h>
  5.  
  6. typedef struct item
  7. {
  8. long arrive; //一位顾客加入队列的时间
  9. int processtime; //该顾客咨询时花费的时间
  10. } Item;
  11.  
  12. #define MAXQUEUE 10
  13.  
  14. typedef struct node
  15. {
  16. Item item;
  17. struct node * next;
  18. } Node;
  19.  
  20. typedef struct queue
  21. {
  22. Node * front; /* 指向队列首项的指针 */
  23. Node * rear; /* 指向队列尾项的指针 */
  24. int items; /* 队列中的项数 */
  25. } Queue;
  26.  
  27. /* 操作: 初始化队列 */
  28. /* 前提条件: pq指向一个队列 */
  29. /* 后置条件: 队列被初始化 */
  30. void InitializeQueue(Queue * pq);
  31.  
  32. /* 操作: 检查队列是否已满 */
  33. /* 前提条件: pq指向之前被初始化的队列 */
  34. /* 后置条件: 如果队列已满则返回true,否则返回false */
  35. bool QueueIsFull(const Queue * pq);
  36.  
  37. /* 操作: 检查队列是否为空 */
  38. /* 前提条件: pq指向之前被初始化的队列 */
  39. /* 后置条件: 如果队列为空则返回true,否则返回false */
  40. bool QueueIsEmpty(const Queue * pq);
  41.  
  42. /* 操作: 确定队列中的项数 */
  43. /* 前提条件: pq指向之前被初始化的队列 */
  44. /* 后置条件: 返回队列中的项数 */
  45. int QueueItemCount(const Queue * pq);
  46.  
  47. /* 操作: 在队列末尾添加项 */
  48. /* 前提条件: pq指向之前被初始化的队列 */
  49. /* item是要被添加在队列末尾的项 */
  50. /* 后置条件: 如果队列不为空,item将被添加在队列的末尾*/
  51. /* 该函数返回true;否则,队列不变,该函数返回false*/
  52. bool EnQueue(Item item, Queue * pq);
  53.  
  54. /* 操作: 从队列的开头删除项 */
  55. /* 前提条件: pq指向之前被初始化的队列 */
  56. /* 后置条件: 如果队列不为空,队列首端的item将被拷贝到*pitem中 */
  57. /* 如果该操作使得队列为空,则重置队列为空 */
  58. /* 如果队列在操作前为空,该函数返回false */
  59. bool DeQueue(Item * pitem, Queue * pq);
  60.  
  61. /* 操作: 清空队列 */
  62. /* 前提条件: pq指向之前被初始化的队列 */
  63. /* 后置条件: 队列被清空 */
  64. void EmptyTheQueue(Queue * pq);
  65.  
  66. #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()、rand()、srand()原型
  5. #include <stdlib.h>
  6. #include <time.h>
  7. //提供strcpy()原型
  8. #include <string.h>
  9. //提供CHAR_BIT的定义,CHAR_BIT表示每字节的位数
  10. #include <limits.h>
  11. //C99定义了bool、true、false
  12. //如果编译器不支持bool,可以用枚举替代enum bool {false, true};
  13. #include <stdbool.h>
  14. #include "queue.h"
  15.  
  16. #define MIN_PER_HR 60.0
  17.  
  18. bool newcustomer(double x); // 是否有新顾客到来?
  19. Item customertime(long when); // 设置顾客参数
  20.  
  21. int main(int argc, char* argv[])
  22. {
  23. Queue line;
  24. Item temp; // 新的顾客数据
  25. int hours; // 模拟的小时数
  26. int perhour;// 每小时平均多少位顾客
  27. long cycle, cyclelimit; // 循环计数器、计数器的上限
  28. long turnaways = 0; // 因队列已满被拒的顾客数量
  29. long customers = 0; // 加入队列的顾客数量
  30. long served = 0; // 在模拟期间咨询过Sigmund的顾客数量
  31. long sum_line = 0; // 累计的队列总长
  32. int wait_time = 0; // 从当前到Sigmund空闲所需的时间
  33. double min_per_cust;// 顾客到来的平均时间
  34. long line_wait = 0; // 队列累计的等待时间
  35.  
  36. InitializeQueue(&line);
  37. srand((unsigned int)time(0)); // rand() 随机初始化
  38. puts("Case Study: Sigmund Lander's Advice Booth");
  39. puts("Enter the number of simulation hours:");
  40. scanf("%d", &hours);//输入模拟运行的小时数
  41. cyclelimit = MIN_PER_HR * hours;
  42. puts("Enter the average number of customers per hour:");
  43. scanf("%d", &perhour);//每小时平均有多少位顾客
  44. min_per_cust = MIN_PER_HR / perhour;
  45.  
  46. for (cycle = 0; cycle < cyclelimit; cycle++)
  47. {
  48. if (newcustomer(min_per_cust))
  49. {
  50. if (QueueIsFull(&line))
  51. turnaways++;
  52. else
  53. {
  54. customers++;
  55. temp = customertime(cycle);
  56. EnQueue(temp, &line);
  57. }
  58. }
  59.  
  60. if (wait_time <= 0 && !QueueIsEmpty(&line))
  61. {
  62. DeQueue(&temp, &line);
  63. wait_time = temp.processtime;
  64. line_wait += cycle - temp.arrive;
  65. served++;
  66. }
  67.  
  68. if (wait_time > 0)
  69. wait_time--;
  70.  
  71. sum_line += QueueItemCount(&line);
  72. }
  73.  
  74. if (customers > 0)
  75. {
  76. printf("customers accepted: %ld\n", customers);
  77. printf(" customers served: %ld\n", served);
  78. printf(" turnaways: %ld\n", turnaways);
  79. printf("average queue size: %.2f\n",
  80. (double) sum_line / cyclelimit);
  81. printf(" average wait time: %.2f minutes\n",
  82. (double) line_wait / served);
  83. }
  84. else
  85. puts("No customers!");
  86.  
  87. EmptyTheQueue(&line);
  88. puts("Bye!");
  89.  
  90. system("pause");
  91. return 0;
  92. }
  93.  
  94. // x是顾客到来的平均时间(单位:分钟)
  95. // 如果1分钟内有顾客到来,则返回true
  96. bool newcustomer(double x)
  97. {
  98. if (rand() * x / RAND_MAX < 1)
  99. return true;
  100. else
  101. return false;
  102. }
  103.  
  104. // when是顾客到来的时间
  105. // 该函数返回一个Item结构,该顾客到达的时间设置为when,
  106. // 咨询时间设置为1~3的随机值
  107. Item customertime(long when)
  108. {
  109. Item cust;
  110.  
  111. cust.processtime = rand() % 3 + 1;
  112. cust.arrive = when;
  113.  
  114. return cust;
  115. }

运行测试

111.png

标签: C语言

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号