隊列 - C語言實現(摘自數據結構與算法分析 C語言描述)

1、概述:

像棧同樣,隊列(queue)也是表。然而,使用隊列時插入在一端進行而刪除在另外一端進行。算法

隊列的基本操做是Enqueue(入隊),它是在表的末端(叫作隊尾(rear)插入一個元素,還有Dequeue(出隊),它是刪除(或返回)在表的開頭(叫作隊頭(front)的元素。如圖1所示顯示一個隊列的抽象模型。數組


圖1 隊列模型數據結構

2、實現

如同棧的情形同樣,對於隊列而言任何表的實現都是合法的。像棧同樣,對於每一種操做,鏈表實現和數組實現都給出快速的O(1)運行時間。

1. 隊列的鏈表實現

文件名:queue_list.h
  1. #ifndef _QUEUE_LIST_H  
  2. #define _QUEUE_LIST_H  
  3.   
  4. #define ElementType int  
  5. struct Node;  
  6. struct QNode;  
  7. typedef struct Node *PtrToNode;  
  8. typedef PtrToNode Queue;  
  9.   
  10.   
  11. int IsEmpty( Queue Q );  
  12. Queue CreateQueue( void );  
  13. void DisposeQueue( Queue Q );  
  14. void MakeEmpty( Queue Q );  
  15. void Enqueue( ElementType X, Queue Q );  
  16. ElementType Front( Queue Q );  
  17. void Dequeue( Queue Q );  
  18. ElementType FrontAndDequeue( Queue Q );  
  19.   
  20. #endif /* _QUEUE_LIST_H */  
<strong><span style="font-size:14px;">#ifndef _QUEUE_LIST_H
#define _QUEUE_LIST_H

#define ElementType int
struct Node;
struct QNode;
typedef struct Node *PtrToNode;
typedef PtrToNode Queue;


int IsEmpty( Queue Q );
Queue CreateQueue( void );
void DisposeQueue( Queue Q );
void MakeEmpty( Queue Q );
void Enqueue( ElementType X, Queue Q );
ElementType Front( Queue Q );
void Dequeue( Queue Q );
ElementType FrontAndDequeue( Queue Q );

#endif /* _QUEUE_LIST_H */</span></strong>
文件名:queue_list.c
  1. #include "queue_list.h"  
  2. #include "fatal.h"  
  3.   
  4. typedef struct QNode  
  5. {  
  6.   ElementType Element;  
  7.   struct QNode *Next;  
  8. }QNode, *QNodePtr;  
  9.   
  10. struct Node {  
  11.   QNodePtr Front;  
  12.   QNodePtr Rear;  
  13. };  
  14.   
  15. int  
  16. IsEmpty( Queue Q )  
  17. {  
  18.   return Q->Front == Q->Rear;  
  19. }  
  20.   
  21. Queue  
  22. CreateQueue( void )  
  23. {  
  24.   Queue Q;  
  25.   Q = malloc( sizeofstruct Node ) );  
  26.   Q->Front = Q->Rear = malloc( sizeofstruct QNode ) );  
  27.   if (!Q->Front)  
  28.     FatalError( "Out of space!!!");  
  29.   Q->Front->Next = NULL;  
  30.   return Q;  
  31. }  
  32.   
  33. void  
  34. MakeEmpty( Queue Q )  
  35. {  
  36.   if( Q == NULL )  
  37.     Error( "Must use CreateQueue first" );  
  38.   else  
  39.     while( !IsEmpty( Q ) )  
  40.       Dequeue( Q );  
  41. }  
  42.   
  43. void  
  44. DisposeQueue( Queue Q )  
  45. {  
  46.   while( Q->Front ) {  
  47.     Q->Rear = Q->Front->Next;  
  48.     free( Q->Front );  
  49.     Q->Front = Q->Rear;  
  50.   }  
  51.   printf( "\nDispose queue completed!!!" );  
  52. }  
  53. void  
  54. Enqueue( ElementType X, Queue Q )  
  55. {  
  56.   QNodePtr p;  
  57.   p = malloc( sizeof( QNode ) );  
  58.   if (!p)  
  59.     FatalError( "Out of space!!!" );  
  60.   p->Element = X;  
  61.   p->Next = NULL;  
  62.   Q->Rear->Next = p;  
  63.   Q->Rear = p;  
  64. }  
  65.   
  66. ElementType  
  67. Front( Queue Q )  
  68. {  
  69.   if ( !IsEmpty( Q ) )  
  70.     return Q->Front->Next->Element;  
  71.   return 0; /* Return value used to avoid warning */  
  72. }  
  73.   
  74. void  
  75. Dequeue( Queue Q )  
  76. {  
  77.   if ( !IsEmpty( Q ) )  
  78.   {  
  79.     QNodePtr p;  
  80.     p = malloc( sizeof( QNode ) );  
  81.     if (!p)  
  82.       FatalError( "Out of space!!!" );  
  83.     p = Q->Front->Next;  
  84.     Q->Front->Next = p->Next;  
  85.     if ( Q->Rear == p )  
  86.       Q->Rear = Q->Front;  
  87.     free( p );  
  88.   }  
  89. }  
  90.   
  91. ElementType  
  92. FrontAndDequeue( Queue Q )  
  93. {  
  94.   if ( !IsEmpty( Q ) )  
  95.   {  
  96.     QNodePtr p;  
  97.     p = malloc( sizeof( QNode ) );  
  98.     if (!p)  
  99.       FatalError( "Out of space!!!" );  
  100.     p = Q->Front->Next;  
  101.     ElementType temp = 0;  
  102.     temp = p->Element;  
  103.     Q->Front->Next = p->Next;  
  104.     if ( Q->Rear == p )  
  105.       Q->Rear = Q->Front;  
  106.     free( p );  
  107.     return temp;  
  108.   }  
  109.   Error( "Empty queue!!!" );  
  110.   return 0; /* Return value used to avoid warning */  
  111. }  
<strong><span style="font-size:14px;">#include "queue_list.h"
#include "fatal.h"

typedef struct QNode
{
  ElementType Element;
  struct QNode *Next;
}QNode, *QNodePtr;

struct Node {
  QNodePtr Front;
  QNodePtr Rear;
};

int
IsEmpty( Queue Q )
{
  return Q->Front == Q->Rear;
}

Queue
CreateQueue( void )
{
  Queue Q;
  Q = malloc( sizeof( struct Node ) );
  Q->Front = Q->Rear = malloc( sizeof( struct QNode ) );
  if (!Q->Front)
    FatalError( "Out of space!!!");
  Q->Front->Next = NULL;
  return Q;
}

void
MakeEmpty( Queue Q )
{
  if( Q == NULL )
    Error( "Must use CreateQueue first" );
  else
    while( !IsEmpty( Q ) )
      Dequeue( Q );
}

void
DisposeQueue( Queue Q )
{
  while( Q->Front ) {
    Q->Rear = Q->Front->Next;
    free( Q->Front );
    Q->Front = Q->Rear;
  }
  printf( "\nDispose queue completed!!!" );
}
void
Enqueue( ElementType X, Queue Q )
{
  QNodePtr p;
  p = malloc( sizeof( QNode ) );
  if (!p)
    FatalError( "Out of space!!!" );
  p->Element = X;
  p->Next = NULL;
  Q->Rear->Next = p;
  Q->Rear = p;
}

ElementType
Front( Queue Q )
{
  if ( !IsEmpty( Q ) )
    return Q->Front->Next->Element;
  return 0; /* Return value used to avoid warning */
}

void
Dequeue( Queue Q )
{
  if ( !IsEmpty( Q ) )
  {
    QNodePtr p;
    p = malloc( sizeof( QNode ) );
    if (!p)
      FatalError( "Out of space!!!" );
    p = Q->Front->Next;
    Q->Front->Next = p->Next;
    if ( Q->Rear == p )
      Q->Rear = Q->Front;
    free( p );
  }
}

ElementType
FrontAndDequeue( Queue Q )
{
  if ( !IsEmpty( Q ) )
  {
    QNodePtr p;
    p = malloc( sizeof( QNode ) );
    if (!p)
      FatalError( "Out of space!!!" );
    p = Q->Front->Next;
    ElementType temp = 0;
    temp = p->Element;
    Q->Front->Next = p->Next;
    if ( Q->Rear == p )
      Q->Rear = Q->Front;
    free( p );
    return temp;
  }
  Error( "Empty queue!!!" );
  return 0; /* Return value used to avoid warning */
}</span></strong>
文件名:queue_list_main.c
  1. #include <stdio.h>  
  2. #include "queue_list.h"  
  3.   
  4. int main()  
  5. {  
  6.   int n, num, m;  
  7.   int i;  
  8.   Queue Q = CreateQueue();  
  9.   printf( "Initialization complete.\n" );  
  10.   if ( IsEmpty( Q ) )  
  11.     printf( "Queue is empty \n" );  
  12.   printf( "Please input the number of elements in the queue:" );  
  13.   scanf( "%d", &n );  
  14.   printf( "Please input %d elements put in queue:", n );  
  15.   for(i = 0; i < n; i++ )  
  16.   {  
  17.     scanf( "%d", &num );  
  18.     Enqueue( num, Q );  
  19.   }  
  20.   printf( "Front element: %d.\n", Front( Q ) );  
  21.   printf( "Elements in queue:" );  
  22.   while ( !IsEmpty( Q )) {  
  23.     printf( "%3d", FrontAndDequeue( Q ) );  
  24.   }  
  25.   DisposeQueue( Q );  
  26.   printf( "\n" );  
  27.   return 0;  
  28. }  
<strong><span style="font-size:14px;">#include <stdio.h>
#include "queue_list.h"

int main()
{
  int n, num, m;
  int i;
  Queue Q = CreateQueue();
  printf( "Initialization complete.\n" );
  if ( IsEmpty( Q ) )
    printf( "Queue is empty \n" );
  printf( "Please input the number of elements in the queue:" );
  scanf( "%d", &n );
  printf( "Please input %d elements put in queue:", n );
  for(i = 0; i < n; i++ )
  {
    scanf( "%d", &num );
    Enqueue( num, Q );
  }
  printf( "Front element: %d.\n", Front( Q ) );
  printf( "Elements in queue:" );
  while ( !IsEmpty( Q )) {
    printf( "%3d", FrontAndDequeue( Q ) );
  }
  DisposeQueue( Q );
  printf( "\n" );
  return 0;
}</span></strong>

2. 隊列的數組實現

文件名:queue_array.h
  1. #ifndef QUEUE_ARRAY_H  
  2. #define QUEUE_ARRAY_H  
  3.   
  4. #define ElementType int  
  5. struct QueueRecord;  
  6. typedef struct QueueRecord *Queue;  
  7.   
  8. int IsEmpty( Queue Q );  
  9. int IsFull( Queue Q );  
  10. Queue CreateQueue( int MaxElements );  
  11. void DisposeQueue( Queue Q );  
  12. void MakeEmpty( Queue Q );  
  13. void Enqueue( ElementType X, Queue Q );  
  14. ElementType Front( Queue Q );  
  15. void Dequeue( Queue Q );  
  16. ElementType FrontAndDequeue( Queue Q );  
  17.   
  18. #endif /* QUEUE_ARRAY_H */  
<strong><span style="font-size:14px;">#ifndef QUEUE_ARRAY_H
#define QUEUE_ARRAY_H

#define ElementType int
struct QueueRecord;
typedef struct QueueRecord *Queue;

int IsEmpty( Queue Q );
int IsFull( Queue Q );
Queue CreateQueue( int MaxElements );
void DisposeQueue( Queue Q );
void MakeEmpty( Queue Q );
void Enqueue( ElementType X, Queue Q );
ElementType Front( Queue Q );
void Dequeue( Queue Q );
ElementType FrontAndDequeue( Queue Q );

#endif /* QUEUE_ARRAY_H */</span></strong>
文件名:queue_array.c
  1. #include "queue_array.h"  
  2. #include "fatal.h"  
  3.   
  4. /* Place in implementation file 
  5.  * Queue implementation is a dynamically allocated array*/  
  6. #define MinQueueSize ( 5 )  
  7.   
  8. struct QueueRecord  
  9. {  
  10.   int Capacity;  
  11.   int Front;  
  12.   int Rear;  
  13.   int Size;  
  14.   ElementType *Array;  
  15. };  
  16.   
  17. int  
  18. IsEmpty( Queue Q )  
  19. {  
  20.   return Q->Size == 0;  
  21. }  
  22.   
  23. void  
  24. MakeEmpty( Queue Q )  
  25. {  
  26.   Q->Size = 0;  
  27.   Q->Front = 1;  
  28.   Q->Rear = 0;  
  29. }  
  30.   
  31. static int  
  32. Succ( int value, Queue Q )  
  33. {  
  34.   if ( ++value == Q->Capacity )  
  35.     value = 0;  
  36.   return value;  
  37. }  
  38.   
  39. void  
  40. Enqueue( ElementType X, Queue Q )  
  41. {  
  42.   if( IsFull( Q ) )  
  43.     Error( "Full queue" );  
  44.   else  
  45.   {  
  46.     Q->Size++;  
  47.     Q->Rear = Succ( Q->Rear, Q );  
  48.     Q->Array[ Q-> Rear ] = X;  
  49.   }  
  50. }  
  51.   
  52. void  
  53. Dequeue( Queue Q )  
  54. {  
  55.   if ( IsEmpty( Q ) )  
  56.     Error( "Empty queue" );  
  57.   else  
  58.   {  
  59.     Q->Size--;  
  60.     Q->Front++;  
  61.   }  
  62. }  
  63.   
  64. ElementType  
  65. FrontAndDequeue( Queue Q )  
  66. {  
  67.   ElementType temp = 0;  
  68.   if ( IsEmpty( Q ) )  
  69.     Error( "Empty queue" );  
  70.   else  
  71.   {  
  72.     Q->Size--;  
  73.     temp= Q->Array[ Q->Front ];  
  74.     Q->Front = Succ( Q->Front, Q );  
  75.   }  
  76.   return temp;  
  77. }  
  78.   
  79. ElementType  
  80. Front( Queue Q )  
  81. {  
  82.   if ( !IsEmpty( Q ) )  
  83.     return Q->Array[ Q->Front ];  
  84.   Error( "Empty queue" );  
  85.   return 0;  /* Return value used to avoid warning */  
  86. }  
  87.   
  88. int  
  89. IsFull( Queue Q )  
  90. {  
  91.   return Q->Size == Q->Capacity;  
  92. }  
  93.   
  94. Queue  
  95. CreateQueue(int MaxElements )  
  96. {  
  97.   Queue Q;  
  98.   
  99.   if ( MaxElements < MinQueueSize )  
  100.     Error( "Queue size is too small!!!" );  
  101.   Q = malloc( sizeofstruct QueueRecord ) );  
  102.   if ( Q == NULL )  
  103.     FatalError( "Out of space!!!" );  
  104.   Q->Array = malloc( sizeof( ElementType ) * MaxElements );  
  105.   if ( Q->Array == NULL )  
  106.     FatalError( "Out of Space!!!" );  
  107.   Q->Capacity = MaxElements;  
  108.   MakeEmpty( Q );  
  109.   
  110.   return Q;  
  111. }  
  112.   
  113. void  
  114. DisposeQueue( Queue Q )  
  115. {  
  116.   if ( Q != NULL )  
  117.   {  
  118.     free( Q->Array );  
  119.     free( Q );  
  120.   }  
  121. }  
<strong><span style="font-size:14px;">#include "queue_array.h"
#include "fatal.h"

/* Place in implementation file
 * Queue implementation is a dynamically allocated array*/
#define MinQueueSize ( 5 )

struct QueueRecord
{
  int Capacity;
  int Front;
  int Rear;
  int Size;
  ElementType *Array;
};

int
IsEmpty( Queue Q )
{
  return Q->Size == 0;
}

void
MakeEmpty( Queue Q )
{
  Q->Size = 0;
  Q->Front = 1;
  Q->Rear = 0;
}

static int
Succ( int value, Queue Q )
{
  if ( ++value == Q->Capacity )
    value = 0;
  return value;
}

void
Enqueue( ElementType X, Queue Q )
{
  if( IsFull( Q ) )
    Error( "Full queue" );
  else
  {
    Q->Size++;
    Q->Rear = Succ( Q->Rear, Q );
    Q->Array[ Q-> Rear ] = X;
  }
}

void
Dequeue( Queue Q )
{
  if ( IsEmpty( Q ) )
    Error( "Empty queue" );
  else
  {
    Q->Size--;
    Q->Front++;
  }
}

ElementType
FrontAndDequeue( Queue Q )
{
  ElementType temp = 0;
  if ( IsEmpty( Q ) )
    Error( "Empty queue" );
  else
  {
    Q->Size--;
    temp= Q->Array[ Q->Front ];
    Q->Front = Succ( Q->Front, Q );
  }
  return temp;
}

ElementType
Front( Queue Q )
{
  if ( !IsEmpty( Q ) )
    return Q->Array[ Q->Front ];
  Error( "Empty queue" );
  return 0;  /* Return value used to avoid warning */
}

int
IsFull( Queue Q )
{
  return Q->Size == Q->Capacity;
}

Queue
CreateQueue(int MaxElements )
{
  Queue Q;

  if ( MaxElements < MinQueueSize )
    Error( "Queue size is too small!!!" );
  Q = malloc( sizeof( struct QueueRecord ) );
  if ( Q == NULL )
    FatalError( "Out of space!!!" );
  Q->Array = malloc( sizeof( ElementType ) * MaxElements );
  if ( Q->Array == NULL )
    FatalError( "Out of Space!!!" );
  Q->Capacity = MaxElements;
  MakeEmpty( Q );

  return Q;
}

void
DisposeQueue( Queue Q )
{
  if ( Q != NULL )
  {
    free( Q->Array );
    free( Q );
  }
}</span></strong>
文件名:queue_main.c
  1. #include <stdio.h>  
  2. #include "queue_array.h"  
  3.   
  4. int  
  5. main()  
  6. {  
  7.   int size;  
  8.   printf( "Please input queue size: " );  
  9.   scanf( "%d", &size );  
  10.   Queue Q = CreateQueue( size );  
  11.   ElementType temp;  
  12.   printf( "Please input queue elements(seperate by space): " );  
  13.   while ( !IsFull( Q ) )  
  14.   {  
  15.     scanf( "%d", &temp );  
  16.     Enqueue( temp, Q );  
  17.   }  
  18.   
  19.   printf( "Dequeue queue in turn: " );  
  20.   while ( !IsEmpty( Q ) )  
  21.   {  
  22.     printf( "%3d", FrontAndDequeue( Q ) );  
  23.   }  
  24.   printf( "\n" );  
  25.   DisposeQueue( Q );  
  26.   return 0;  
  27. }  
<strong><span style="font-size:14px;">#include <stdio.h>
#include "queue_array.h"

int
main()
{
  int size;
  printf( "Please input queue size: " );
  scanf( "%d", &size );
  Queue Q = CreateQueue( size );
  ElementType temp;
  printf( "Please input queue elements(seperate by space): " );
  while ( !IsFull( Q ) )
  {
    scanf( "%d", &temp );
    Enqueue( temp, Q );
  }

  printf( "Dequeue queue in turn: " );
  while ( !IsEmpty( Q ) )
  {
    printf( "%3d", FrontAndDequeue( Q ) );
  }
  printf( "\n" );
  DisposeQueue( Q );
  return 0;
}</span></strong>

附錄:上述代碼中用到了Error、FatalError等函數,其實現以下(即fatal.h文件):
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3.   
  4. #define Error( Str )        FatalError( Str )  
  5. #define FatalError( Str )   fprintf( stderr, "%s\n", Str ), exit( 1 )  
<strong><span style="font-size:14px;">#include <stdio.h>
#include <stdlib.h>

#define Error( Str )        FatalError( Str )
#define FatalError( Str )   fprintf( stderr, "%s\n", Str ), exit( 1 )
</span></strong>

備註:本文摘自《數據結構與算法分析 C語言描述 Mark Allen Weiss著》,代碼經gcc編譯測試經過。

附件下載:http://download.csdn.net/detail/shuxiao9058/4212437#queue-array.tar.gz,http://download.csdn.net/detail/shuxiao9058/4212435#queue-list.tar.gz函數


相關文章
相關標籤/搜索