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

发布者:紫色小猫最新更新时间:2017-01-15 来源: eefocus关键字:单片机  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;


#include "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

//////////////////////////////////////////////////////////

#include "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

//////////////////////////////////////////////////////////

#include

#include "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)展示了这两种操作进行时指针变化的情况。下面给出链队列类型的模块说明。

          图3.10  

                                           图2 链队列示意图       

                    图3.11

       图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;否则返回ERROR 

Status 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所示。

                 图3.12

                                             图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无法判别队列空间是“空”还是“满”。可由两种处理方法:其一是另设一个标志位以区别队列是“空”还是“满”;其二是少用一个元素空间,约定以“队列头指针在队列尾指针的下一位置(指环状的下一位置)上”作为队列呈“满”状态的标志。

                                           图3.13

 

                                                                    图5 循环队列示意图

                                 图3.14

                             图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; 

}


关键字:单片机  FIFO  循环队列 引用地址:单片机的FIFO(先入先出)循环队列实现

上一篇:单片机输出C#点灯神话
下一篇:串口多字节接收

推荐阅读最新更新时间:2024-03-16 15:30

初步认识51单片机-2.2单片机控制LCD1602液晶显示模块
上面学的两招,控制IO和延时,在这里要举的第一个例子就是LCD1602。LCD1602什么意思,表示一行可以显示16个字符,一共有两行。先来个LCD1602的简单介绍,1602LCD主要技术参数: 显示容量:16 2个字符 芯片工作电压:4.5 5.5V 工作电流:2.0mA(5.0V) 模块最佳工作电压:5.0V 字符尺寸:2.95 4.35(W H)mm 引脚功能说明 1602LCD采用标准的14脚(无背光)或16脚(带背光)接口,各引脚接口说明如表1所示: 第1脚:VSS为地电源。 第2脚:VDD接5V正电源。 第3脚:VL为液晶显示器对比度调整端,接正电源时对比度最弱,接地时对比度最高,对比度过高时会产
[单片机]
初步认识51<font color='red'>单片机</font>-2.2<font color='red'>单片机</font>控制LCD1602液晶显示模块
基于AT91M42800A的LED显示系统设计
引 言 :   最近,笔者在某工厂大型生产线上基于现场总线的物流呼叫系统项目中发现,由于所需要显示的信息流比较大,用现有的基于AT89C51芯片组成的LED显示屏控制系统,由于受到微处理器的处理速度、体系架构、寻址范围、外围接口资源等诸多限制,已难以在要求显示较多像素、显示内容帧频较高、动态显示效果复杂的情况下,得到良好的动态视觉效果。针对以上情况,在利用现有资源的基础上,重新设计和研制了一种全新的,由32位高性能ARM微处理器组成的LED显示屏控制图1系统的硬件结构框图系统,并通过RS485接口与现场总线中的上位机进行实时数据通信,实现整个系统的信息显示。 1 系统硬件结构   该系统的硬件组成框图如图1所示。图1中,微处
[工业控制]
7脚12864spi单片机源程序,直接函数调用
单片机源程序如下: #include yejin.h const uchar num ={ 0x00,0xE0,0x10,0x08,0x08,0x10,0xE0,0x00,0x00,0x0F,0x10,0x20,0x20,0x10,0x0F,0x00,/* 0 ,0*/ 0X00,0X00,0X08,0X08,0X1F,0X00,0X00,0X00,0X00,0X00,0X04,0X04,0XFC,0X04,0X04,0X00,/* 1 ,1*/ 0x00,0x70,0x08,0x08,0x08,0x08,0xF0,0x00,0x00,0x30,0x28,0x24,0x22,0x21,0x30,0x00,/* 2 ,
[单片机]
基于C51单片机的计时器设计原理图
如下图所示,在 AT89S51 单片机的 P0 和 P2 端口分别接有两个共阴数码管 P0 口驱动显示秒时间的十位,而 P2 口驱动显示秒时间的个位。   1 . 把 “ 单片机系统 ” 区域中的 P0.0/AD0 - P0.7/AD7 端口用 8 芯排线连接到“ 四路静态数码显示模块 ” 区域中的任一个 a - h 端口上;要求: P0.0/A D0对应着 a , P0.1/AD1 对应着 b , …… , P0.7/AD7 对应着 h 。   2 . 把 “ 单片机系统 ” 区域中的 P2.0/A8 - P2.7/A15 端口用 8 芯排线连接到 “ 四路静态数码显示模块 ” 区域中的任一个 a - h 端口上;要求: P
[模拟电子]
基于C51<font color='red'>单片机</font>的计时器设计原理图
MCU的每一位都恰到好处,是门学问
市场研究机构IC Insights的数据显示,2019年全球微控制器(MCU)营收预计将成长9%,达到204亿美元,并在未来五年以7.2%的年复合增长率高速成长。面对层出不穷的应用需求,MCU厂商该如何主动出击寻求变化?MCU的下一个风口在哪里? 长盛不衰的8位MCU STM8是意法半导体(ST)在十年前设计的产品,但目前来看还是能满足多方面应用需求的。究其原因,ST微控制器市场产品经理PATRICE HAMARD认为一是客户的使用习惯,这些客户一直在使用8位MCU产品;其次是出于成本的控制,很多成本敏感型应用需要更低价的MCU;第三是STM8更简单、易用。 左:ST微控制器市场产品经理PATRICE HA
[嵌入式]
让<font color='red'>MCU</font>的每一位都恰到好处,是门学问
基于单片机和CPLD的数字频率计的设计(图)
引言 在传统的控制系统中,通常将单片机作为控制核心并辅以相应的元器件构成一个整体。但这种方法硬件连线复杂、可靠性差,且在实际应用中往往需要外加扩展芯片,这无疑会增大控制系统的体积,还会增加引入干扰的可能性。对一些体积小的控制系统,要求以尽可能小的器件体积实现尽可能复杂的控制功能,直接应用单片机及其扩展芯片就难以达到所期望的效果。 复杂可编程逻辑器件(CPLD)具有集成度高、运算速度快、开发周期短等特点,它的出现,改变了数字电路的设计方法、增强了设计的灵活性。基于此,本文提出了一种采用Altera公司的CPLD(ATF1508AS)和Atmel公司的单片机(AT89S52)相结合的数字频率计的设计方法。该数字频率计电路简洁,
[电源管理]
基于<font color='red'>单片机</font>和CPLD的数字频率计的设计(图)
32位低功耗MCU的设计
1 前言 传统的低功耗MCU设计都是以8位MCU为主,因为8位内核逻辑门数相对较少,运行或泄露电流低,售价也相对低廉。但是,许多新兴的应用都需要比8位内核更大的处理效率。近年智能生活的抬头、物联网的建立,便携式消费性电子产品与无线功能需求越来越高、设计越来越复杂,要提高性能的同时又要兼顾低功耗,需要有一款高性能低功耗的主控MCU来作为平台。另一方面,工业上的智能化也在展开,如远程监控、数字化、网络化等。简单说来,就是人物连接(云端应用)、物物连接(物联网)需求越来越多,导致产品功能越来越复杂,计算量越来越高,2009年ARM发表了32位Cortex-M0内核,提供给MCU厂商一个强而有力的平台,加上工艺微缩技术的进步,嵌入式闪存工
[单片机]
32位低功耗<font color='red'>MCU</font>的设计
适合单片机裸机的开源软件框架:Zorb
很多时候,做单片机项目,会因为性能和内存资源的限制,没办法运行一些“大型”的通用框架,这个时候,一些轻量级的软件框架有显得尤为重要了。 这里就给大家分享一款一款适合单片机裸机的开源软件框架:Zorb Zorb简介 Zorb Framework是一个基于面向对象的思想来搭建一个轻量级的嵌入式框架。 搭建Zorb Framework的目的是为在不能运行Linux的芯片上快速开发应用,不用反复造轮子。 Zorb Framework的初步设计功能有: 1、时间系统功能zf_time 2、环形缓冲区功能zf_buffer 3、列表功能zf_list 4、状态机功能zf_fsm 5、事件功能zf_event 6、定时器功能zf_time
[单片机]
适合<font color='red'>单片机</font>裸机的开源软件框架:Zorb
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved