单片机实现环形队列_单片机的FIFO(先入先出)循环队列实现

//

// 文件:config.h

//

#ifndef __CONFIG_H #define __CONFIG_H

//这一段无需改动

//This segment should not be modified

#ifndef TRUE

#define TRUE 1

#endif

#ifndef FALSE

#define FALSE 0

#endif

typedef unsigned

char uint8; typedef

signed char int8; typedef unsigned short

uint16; typedef

signed short

int16; typedef unsigned

int uint32; typedef

signed int int32; typedef

float fp32;

#i nclude "FIFOQUEUE.h"

#endif

//

// 文件:FIFOQUEUE.h

//

#ifndef _FIFOQUEUE_H

#define _FIFOQUEUE_H

#define

ElemType uint8

#define

QueueSize 20

//fifo队列的大小

#define

QueueFull 0

//fifo满置0

#define

QueueEmpty 1

//FIFO空置1

#define QueueOperateOk 2  //队列操作完成 赋值为2

struct FifoQueue

{

uint16

front;  //队列头

uint16

rear;  //队列尾

uint16

count;  //队列计数

ElemType

dat[QueueSize];

};

//Queue Initalize

extern void QueueInit(struct FifoQueue *Queue);

// Queue In

extern uint8 QueueIn(struct FifoQueue *Queue,ElemType sdat);

// Queue Out

extern uint8 QueueOut(struct FifoQueue *Queue,ElemType

*sdat);

#endif

//

// 文件:FIFOQUEUE.C

//

#i nclude "config.h"

//Queue Init

void QueueInit(struct FifoQueue *Queue)

{

Queue->front

= Queue->rear;//初始化时队列头队列首相连

Queue->count

= 0;  //队列计数为0

}

// Queue In

uint8 QueueIn(struct FifoQueue *Queue,ElemType sdat) //数据进入队列

{

if((Queue->front

== Queue->rear) &&

(Queue->count == QueueSize))

{ //

full //判断如果队列满了

return

QueueFull;  //返回队列满的标志

}else

{ //

in

Queue->dat[Queue->rear]

= sdat;

Queue->rear

= (Queue->rear + 1) % QueueSize;

Queue->count

= Queue->count + 1;

return

QueueOperateOk;

}

}

// Queue Out

uint8 QueueOut(struct FifoQueue *Queue,ElemType *sdat)

{

if((Queue->front

== Queue->rear) &&

(Queue->count == 0))

{ //

empty

return

QueueEmpty;

}else

{ //

out

*sdat

= Queue->dat[Queue->front];

Queue->front

= (Queue->front + 1) % QueueSize;

Queue->count

= Queue->count - 1;

return

QueueOperateOk;

}

}

//

// 文件:Main.C

//

#i nclude

#i nclude "config.h"

void main(void)

{

struct

FifoQueue MyQueue;

ElemType

sh;

uint8

i;

QueueInit(&MyQueue);

while(1)

{

for(i

= 0;i < 30;i++)

{

if(QueueIn(&MyQueue,i)

== QueueFull) break;

}

for(i

= 0;i < 30;i++)

{

if(QueueOut(&MyQueue,&sh)

== QueueEmpty) break;

}

}

while(1);

}

队列是一种先进先出(first infirst

out,缩写为FIFO)的线性表。它只允许在标的一端进行插入,而在另一端删除元素。这和我们日常生活中的排队是一致的,最早进入队列的元素最早离开。在队列中,允许插入的一端叫做队尾(rear),允许删除的一端则称为对头(front)(排队买票,窗口一端叫对头,末尾进队叫队尾)。

用链表表示的队列称为链队列,如图2所示。一个链队列显然需要两个分别指向对头和队尾的指针(分别称为头指针和尾指针)才能唯一确定。这里,和线性表的单链表一样,为了操作方便起见,我们先给链队列添加一个头结点,并令头指针和尾指针均指向头结点,如图3(a)所示。链队列的操作即为单链表的插入和删除操作的特殊情况,只是尚需修改尾指针或头指针,图3(b)~(d)展示了这两种操作进行时指针变化的情况。下面给出链队列类型的模块说明。

 

图2

链队列示意图 图3 队列运算指针变化情况 (a)空队列;(b)元素x入队;(c)元素y入队;(d)元素x出队

//=====ADT

Queue的表示与实现===== //-----单链队列——队列的链式存储结构-----typedef

struct QNode{ QElemType data; struct

QNode *next; }QNode, *QueuePtr;

typedef struct{ QueuePtr

front; //对头指针 QueuePtr rear; //队尾指针}LinkQueue;

//-----基本操作的函数原型说明(几个易错常考的)-----Status GetHead(LinkQueue Q, QElemType

&e) //若队列不空,则用e返回Q的对头元素,并返回OK;否则返回ERRORStatus EnQueue(LinkQueue &Q, QElemType

e) //插入元素e为Q的新的队尾元素Status DeQueue(LinkQueue &Q, QElemType

&e) //若队列不空,则删除Q的对头元素,用e返回其值,并返回OK; //否则返回ERROR

和顺序栈相类似,在队列的顺序存储结构中,除了用一组地址连续的存储单元依次存放从队列头到队列尾的元素之外,尚需附设两个指针front和rear分别之时队列头元素和队列尾元素的位置。为了在C语言中描述方便起见,在此我们约定:初始化建空队列时,令front=rear=0,每当插入新的队列尾元素时,“尾指针增1”;每当删除队列头元素时,“头指针增1”。因此,在非空队列中,头指针始终指向队列头元素,而尾指针始终指向队列尾元素的下一个位置。如图4所示。

图4

头、尾指针和队列中元素之间的关系

(a)空队列;(b)J1、J2和J3相继入队列;(c)J1和J2相继被删除;(d)J4、J5和J6相继插入队列之后J3及J4被删除

假设当前为队列分配的最大空间为6,则当队列处于图4(d)的状态时不可再继续插入新的队尾元素,否则会因数组越界而遭致程序代码被破坏。然而此时又不宜如顺序栈那样,进行存储再分配扩大数组空间,因为队列的实际可用空间并未占满。一个较巧妙的办法是将顺序队列臆造为一个环状的空间,如图5所示,称之为循环队列。指针和队列元素之间关系不变,如图6(a)所示循环队列中,队列头元素时J3,队列尾元素是J5,之后J6、J7和J8相继插入,则队列空间均被占满,如图6(b)所示,此时Q.front=Q.rear;反之,若J3、J4和J5相继从图6(a)的队列中删除,使队列呈“空”的状态,如图6(c)所示。此时亦存在关系式Q.front=Q.rear,由此可见,只凭等式Q.front=Q.rear无法判别队列空间是“空”还是“满”。可由两种处理方法:其一是另设一个标志位以区别队列是“空”还是“满”;其二是少用一个元素空间,约定以“队列头指针在队列尾指针的下一位置(指环状的下一位置)上”作为队列呈“满”状态的标志。

图5 循环队列示意图

图6 循环队列的头尾指针

(a)一般情况;(b)队列满时;(c)空队列;

从上述分析可见,在C语言中不能用动态分配的一维数组来实现循环队列。如果用户的应用程序中设有循环队列,则必须为它设定一个最大队列长度;若用户无法预估所用队列的最大长度,则宜采用链队列。循环队列类型的模块说明如下:

//-----循环队列———队列的顺序存储结构-----#define MAXQSIZE 100 //最大队列长度 typedef struct{ QElemType

*base; //初始化的动态非配存储空间 int

front; //头指针,若队列不空,指向队列的头元素 int

rear; //尾指针,若队列不空,指向队列尾元素的下一个位置}SqQueue;

//-----循环队列的基本操作的算法描述-----Status InitQueue(SqQueue

&Q){ //构造一个空队列Q Q.base=(ElemType

*)malloc(MAXQSIZE*sizeof(ElemType)); if(!Q.base)

exit (OVERFLOW); //

存储分配失败 Q.front=Q.rear=0; return

OK; }

int QueueLength(SqQueue Q){ //返回Q的元素个数,即队列的长度 return (Q.rear-Q.front+MAXQSIZE) %

MAXQSIZE; }

Status EnQueue(SqQueue &Q, QElemType

e){ //插入元素e为Q的新的队尾元素 if((Q.rear+1) % MAXQSIZE == Q.front) return

ERROR; //

队列满 Q.base[Q.rear]=e; Q.rear=(Q.rear+1) % MAXQSIZE; return

OK; }

Status DeQueue(SqQueue &Q, QElemType

&e){ //若队列不空,则删除Q的对头元素,用e返回其值,并返回OK; //否则返回ERROR if(Q.front==Q.rear) return

ERROR; e=Q.base[Q.front]; Q.front=(Q.front+1) % MAXQSIZE; return

OK; }

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值