0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

单片机消息队列的实现原理和机制

strongerHuang 来源: strongerHuang 2023-05-26 09:50 次阅读

单片机开发过程中通常会用到“消息队列”,一般实现的方法有多种。 本文给大家分享一下队列实现的原理和机制。

环形队列

环形队列是在实际编程极为有用的数据结构,它是一个首尾相连的FIFO的数据结构,采用数组的线性空间,数据组织简单,能很快知道队列是否满为空,能以很快速度的来存取数据。

环形队列通常用于通信领域,比如UARTUSBCAN网络等。

1.环形队列实现原理

内存上没有环形的结构,因此环形队列实上是数组的线性空间来实现。 当数据到了尾部它将转回到0位置来处理。

因此环列队列的逻辑:将数组元素q[0]与q[MAXN-1]连接,形成一个存放队列的环形空间。

为了方便读写,还要用数组下标来指明队列的读写位置。 head/tail.其中head指向可以读的位置,tail指向可以写的位置。

wKgZomRwEQeAZHOsAAAztrtbyhU714.jpg

环形队列的关键是判断队列为空,还是为满。 当tail追上head时,队列为满时; 当head追上tail时,队列为空。 但如何知道谁追上谁,还需要一些辅助的手段来判断.

如何判断环形队列为空,为满有两种判断方法:

a.附加一个标志位tag

当head赶上tail,队列空,则令tag=0

当tail赶上head,队列满,则令tag=1

b.限制tail赶上head,即队尾结点与队首结点之间至少留有一个元素的空间。

队列空: head==tail

队列满: (tail+1)% MAXN ==head


2.附加标志实现原理

a.采用第一个环形队列有如下结构:

typedef struct ringq{
   int head; /* 头部,出队列方向*/
   int tail; /* 尾部,入队列方向*/ 
   int tag ;
   int size ; /* 队列总尺寸 */
   int space[RINGQ_MAX]; /* 队列空间 */
}RINGQ;
初始化状态:
q->head = q->tail = q->tag = 0;
队列为空:
( q->head == q->tail) && (q->tag == 0)
队列为满 :
 ((q->head == q->tail) && (q->tag == 1))
入队操作,如队列不满,则写入:
q->tail =  (q->tail + 1) % q->size ;
出队操作,如果队列不空,则从head处读出。

下一个可读的位置在:
q->head =  (q->head + 1) % q->size
b.完整代码

头文件ringq.h:
#ifndef __RINGQ_H__
#define __RINGQ_H__


#ifdef __cplusplus
extern "C" {
#endif 


#define QUEUE_MAX 20


typedef struct ringq{
   int head; /* 头部,出队列方向*/
   int tail; /* 尾部,入队列方向*/ 
   int tag ; /* 为空还是为满的标志位*/
    int size ; /* 队列总尺寸 */
   int space[QUEUE_MAX]; /* 队列空间 */
}RINGQ;


/* 
  第一种设计方法:
     当head == tail 时,tag = 0 为空,等于 = 1 为满。
*/


extern int ringq_init(RINGQ * p_queue);


extern int ringq_free(RINGQ * p_queue);




/* 加入数据到队列 */
extern int ringq_push(RINGQ * p_queue,int data);


/* 从队列取数据 */
extern int ringq_poll(RINGQ * p_queue,int *p_data);




#define ringq_is_empty(q) ( (q->head == q->tail) && (q->tag == 0))


#define ringq_is_full(q) ( (q->head == q->tail) && (q->tag == 1))


#define print_ringq(q) printf("ring head %d,tail %d,tag %d
", q->head,q->tail,q->tag);
#ifdef __cplusplus
}
#endif 


#endif /* __RINGQ_H__ */
源代码 ringq.c:
#include 
#include "ringq.h"


int ringq_init(RINGQ * p_queue)
{
   p_queue->size = QUEUE_MAX ;


   p_queue->head = 0;
   p_queue->tail = 0;


   p_queue->tag = 0;


   return 0;
}


int ringq_free(RINGQ * p_queue)
{
  return 0;
}




int ringq_push(RINGQ * p_queue,int data)
{
  print_ringq(p_queue);


  if(ringq_is_full(p_queue))
   {


     printf("ringq is full
");
     return -1;
   }


   p_queue->space[p_queue->tail] = data;


   p_queue->tail = (p_queue->tail + 1) % p_queue->size ;


   /* 这个时候一定队列满了*/
   if(p_queue->tail == p_queue->head)
    {
       p_queue->tag = 1;
    }


    return p_queue->tag ;  
}


int ringq_poll(RINGQ * p_queue,int * p_data)
{
   print_ringq(p_queue);
  if(ringq_is_empty(p_queue))
   {


      printf("ringq is empty
");
     return -1;
   }


   *p_data = p_queue->space[p_queue->head];


   p_queue->head = (p_queue->head + 1) % p_queue->size ;


    /* 这个时候一定队列空了*/
   if(p_queue->tail == p_queue->head)
    {
       p_queue->tag = 0;
    }    
    return p_queue->tag ;
}

看到源代码,相信大家就明白其中原理了。其实还有不采用tag,或者其他一些标志的方法,这里就不进一步展开讲述了,感兴趣的读者可以自行研究一下。

消息队列

RTOS中基本都有消息队列这个组件,也是使用最常见的组件之一。

1.消息队列的基本概念

消息队列是一种常用于任务间通信的数据结构,队列可以在任务与任务间、中断和任务间传递信息,实现了任务接收来自其他任务或中断的不固定长度的消息。

通过消息队列服务,任务或中断服务程序可以将一条或多条消息放入消息队列中。同样,一个或多个任务可以从消息队列中获得消息。

使用消息队列数据结构可以实现任务异步通信工作。

2.消息队列的特性

RTOS消息队列,常见特性:

消息支持先进先出方式排队,支持异步读写工作方式。

读写队列均支持超时机制。

消息支持后进先出方式排队,往队首发送消息(LIFO)。

可以允许不同长度(不超过队列节点最大值)的任意类型消息。

一个任务能够从任意一个消息队列接收和发送消息。

多个任务能够从同一个消息队列接收和发送消息。

当队列使用结束后,可以通过删除队列函数进行删除。

3.消息队列的原理

这里以 FreeRTOS 为例进行说明。FreeRTOS 的消息队列控制块由多个元素组成,当消息队列被创建时,系统会为控制块分配对应的内存空间,用于保存消息队列的一些信息如消息的存储位置,头指针 pcHead、尾指针 pcTail、消息大小 uxItemSize 以及队列长度 uxLength 等

wKgZomRwEOaAAVkKAAUULUbui9o464.png

比如创建消息队列:

xQueue = xQueueCreate(QUEUE_LEN, QUEUE_SIZE);

任务或者中断服务程序都可以给消息队列发送消息,当发送消息时,如果队列未满或者允许覆盖入队,FreeRTOS 会将消息拷贝到消息队列队尾,否则,会根据用户指定的阻塞超时时间进行阻塞,在这段时间中,如果队列一直不允许入队,该任务将保持阻塞状态以等待队列允许入队。 当其它任务从其等待的队列中读取入了数据(队列未满),该任务将自动由阻塞态转移为就绪态。 当等待的时间超过了指定的阻塞时间,即使队列中还不允许入队,任务也会自动从阻塞态转移为就绪态,此时发送消息的任务或者中断程序会收到一个错误码 errQUEUE_FULL。

发送紧急消息的过程与发送消息几乎一样,唯一的不同是,当发送紧急消息时, 发送的位置是消息队列队头而非队尾,这样,接收者就能够优先接收到紧急消息,从而及时进行消息处理。

当某个任务试图读一个队列时,其可以指定一个阻塞超时时间。 在这段时间中,如果队列为空,该任务将保持阻塞状态以等待队列数据有效。 当其它任务或中断服务程序往其等待的队列中写入了数据,该任务将自动由阻塞态转移为就绪态。 当等待的时间超过了指定的阻塞时间,即使队列中尚无有效数据,任务也会自动从阻塞态转移为就绪态。

当消息队列不再被使用时,应该删除它以释放系统资源,一旦操作完成, 消息队列将被永久性的删除。

消息队列的运作过程具体见下图:

wKgaomRwEMuAEr8ZAAD-lTL-UDw319.png

4.消息队列的阻塞机制

出队阻塞:当且仅当消息队列有数据的时候,任务才能读取到数据,可以指定等待数据的阻塞时间。

入队阻塞:当且仅当队列允许入队的时候,发送者才能成功发送消息; 队列中无可用消息空间时,说明消息队列已满,此时,系统会根据用户指定的阻塞超时时间将任务阻塞。

假如有多个任务阻塞在一个消息队列中,那么这些阻塞的任务将按照任务优先级进行排序,优先级高的任务将优先获得队列的访问权。

“环形队列”和“消息队列”的异同

通过以上分析,你会发现“环形队列”和“消息队列”之间有很多共同点:

1.他们都是一种数据结构,结构中都包含头、尾、标志等信息;

2.它们都是分配一块连续的内存空间,且都可以分配多个队列。

3.应用场景类似,有大量吞吐数据的情况下,比如通信领域。

当然,他们也有一些不同点:

1.“环形队列”可以独立使用,也可以结合操作系统使用。 而消息队列依赖RTOS(有些RTOS的参数信息)。

2.“环形队列”占用资源更小,更适合于资源较小的系统中。

3.“消息队列”结合RTOS应用更加灵活,比如延时、中断传输数据等。

最后,这两种队列应用都比较广,建议抽空都研究一下。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 单片机
    +关注

    关注

    6001

    文章

    43973

    浏览量

    620847
  • uart
    +关注

    关注

    22

    文章

    1159

    浏览量

    99961
  • RTOS
    +关注

    关注

    20

    文章

    776

    浏览量

    118796
  • 数据结构
    +关注

    关注

    3

    文章

    564

    浏览量

    39900
  • 消息队列
    +关注

    关注

    0

    文章

    31

    浏览量

    2921

原文标题:单片机消息队列的实现原理和机制

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    单片机的扩展机制

    新手上路,请问就是单片机的存储器扩展机制不是指扩充了物理意义上的内存,而是指所能访问的内容,地址增多了,是不?
    发表于 08-07 11:17

    Small RTOS 的设计思想及消息队列通信机制的应用

    将Small RTOS 多任务的思想应用在一个具体的单片机控制的电子式存包柜系统的软件设计中,介绍了基于消息队列的任务通讯的编程方法。
    发表于 04-23 06:22

    单片机开发的队列该怎样去使用呢

    背景  最近做单片机开发经常遇见要用串口接收数据的情况,实际项目中肯定不能当串口接收中断一来就去处理,于是我们可以用到队列这个数据结构来保存上一帧数据,想用的时候取出即可。开始新建队列结构体
    发表于 11-22 07:45

    如何使用ITM机制实现调试stm32单片机实现printf与scanf?

    如何使用ITM机制实现调试stm32单片机实现printf与scanf?
    发表于 12-02 06:53

    实现队列环形缓冲的方法

    (rear)入队;  队列实现可以用数组(顺序存储结构)和链表(链式存储结构)实现,但使用链表会产生额外的开销(存在指针域),对于单片机这类存储容量有限的微处理器,不是很实用。所以我
    发表于 02-21 07:11

    怎样去使用基于单片机实现队列功能模块QueueForMcu呢

    基于单片机实现队列功能模块QueueForMcu有何特性?怎样去使用基于单片机实现队列功能模
    发表于 02-23 06:39

    51和32单片机的复位机制有哪些不同?

    51和32单片机的复位机制有哪些不同。以及复位电容电压的计算方法。
    发表于 11-06 06:16

    51单片机多任务机制实现策略研究

    从操作系统实现多任务机制的原理入手,分析了51单片机实现多任务机制的基本条件,论述了5l单片机
    发表于 09-19 17:26 159次下载
    51<b class='flag-5'>单片机</b>多任务<b class='flag-5'>机制</b>的<b class='flag-5'>实现</b>策略研究

    单片机制作继电器 【汇编版】

    单片机制作继电器 【汇编版】单片机制作继电器 【汇编版】单片机制作继电器 【汇编版】
    发表于 12-29 13:50 0次下载

    单片机开发——应用消息队列处理事件

    单片机开发——应用消息队列处理事件
    发表于 11-13 13:36 11次下载
    <b class='flag-5'>单片机</b>开发——应用消息<b class='flag-5'>队列</b>处理事件

    单片机串口——队列的使用

    背景  最近做单片机开发经常遇见要用串口接收数据的情况,实际项目中肯定不能当串口接收中断一来就去处理,于是我们可以用到队列这个数据结构来保存上一帧数据,想用的时候取出即可。开始新建队列结构
    发表于 11-13 20:36 16次下载
    <b class='flag-5'>单片机</b>串口——<b class='flag-5'>队列</b>的使用

    QueueForMcu 基于单片机实现队列功能模块

    QueueForMcu基于单片机实现队列功能模块,主要用于8位、16位、32位非运行RTOS的单片机应用,兼容大多数单片机平台。一、特性动
    发表于 12-31 19:35 1次下载
    QueueForMcu 基于<b class='flag-5'>单片机</b><b class='flag-5'>实现</b>的<b class='flag-5'>队列</b>功能模块

    ZWave中的消息队列机制是什么

    这篇文章就来看看 ZWave 中是通过什么机制为我们提供了一个便捷的消息队列处理机制
    的头像 发表于 02-14 13:41 500次阅读
    ZWave中的消息<b class='flag-5'>队列</b><b class='flag-5'>机制</b>是什么

    一种基于单片机实现队列功能模块

    基于单片机实现队列功能模块,主要用于8位、16位、32位非运行RTOS的单片机应用,兼容大多数单片机平台。
    的头像 发表于 08-14 11:09 491次阅读
    一种基于<b class='flag-5'>单片机</b><b class='flag-5'>实现</b>的<b class='flag-5'>队列</b>功能模块

    单片机裸机实现队列功能的方案

    单片机裸机实现队列功能的方案
    的头像 发表于 10-17 14:34 318次阅读