跟着野火学FreeRTOS:第二段(队列管理)

     队列( Q u e u e s Queues Queues)是 F r e e R T O S FreeRTOS FreeRTOS中的一种数据结构,这种数据结构提供了一种任务和任务之间,任务和中断之间的通信机制。队列可以存储一定有限数量的固定大小( u x I t e m S i z e uxItemSize uxItemSize)的数据项,这个 一定有限数量的值就是队列的长度 u x Q u e u e L e n g t h uxQueueLength uxQueueLength u x Q u e u e L e n g t h uxQueueLength uxQueueLength u x I t e m S i z e uxItemSize uxItemSize的值都在是创建队列的时候指定好的,队列除了存储数据的空间以外还有一部分空间用来记录队列的相关信息,比如前面提到的队列的长度以及数据项的大小,该部分信息用一个结构体来定义如图2所示。队列的大致的结构,如图1所示,图1中队列的长度位 n + 1 n+1 n+1。记录队列的相关信息的空间位于分配给队列的空间的最前面,后面接着就是存储队列中实际数据的空间。

图1.
图2.

     图3到图6用图示的形式简单的介绍了一下利用队列进行通信的流程,这里 T a s k B Task\quad B TaskB接收 T a s k A Task\quad A TaskA发送的数据。图3是队列刚刚建立好的时候,队列里面没有通信的数据,图4和图5分别是 T a s k A Task\quad A TaskA向队列里面发送了数据10接着发送了数据20之后的情况,图6是 T a s k B Task\quad B TaskB接收了 T a s k A Task\quad A TaskA第一次发送的数据10之后队列的情况。一般情况下队列是一种 F I F O ( F i r s t I n F i r s t O u t ) FIFO(First\quad In\quad First\quad Out) FIFO(FirstInFirstOut)类型的数据结构,往队列里面发送数据的时候会将数据发送到队列的尾部( T a s k A Task\quad A TaskA第二次写入的数据20位于第一次写入的数据10的后面),从队列里面接收数据的时候会从队列的头部读取数据(在图5中此时队列里面已经有两个数据, T a s k B Task\quad B TaskB此时去读取队列里面数据的时候读取到的是数据10),但是 F r e e R T O S FreeRTOS FreeRTOS也支持往队列里面发送数据的时候将数据发送到队列的头部,这样的话写入头部的数据就可以优先被接收其它接收数据的任务先接收到,这种操作可以用于一些需要紧急处理的数据。

图3.
图4.
图5.
图6.

     还有就是发送到队列里面的数据是直接将要发送的数据 复制一份到图1队列里面的 x I t e m x xItem\quad x xItemx里面,而不是将要发送的数据的指针存储到队列里面。
     同一个队列可以被多个任务或中断服务程序写入数据或读出数据,比较常见的情况是多个任务或中断服务程序对同一个队列进行数据写入,但是多个任务或中断服务程序对同一个队列进行读出数据的情况比较少见。
      F r e e R T O S FreeRTOS FreeRTOS里面的读队列操作支持一个可选的 阻塞时间参数,当这个参数不为0的时候,如果任务在尝试读队列的时候,队列为空,这时任务会进入阻塞状态等待其它任务或中断服务程序往这个队列里面写入数据,如果在参数配置的时间之内有其它任务或中断服务程序往这个队列里面写入了数据,那么这个处于阻塞状态的任务会立即转换为就绪态,假设在参数配置的时间之内没有其它任务或中断服务程序往这个队列里面写入了数据,这个处于阻塞状态的任务也会自动转换为就绪态。假如有多个任务因为尝试读取同一队列(这里多个任务的阻塞时间参数都不为0),但是队列为空,而都进入到了阻塞状态,假如在它们的阻塞时间参数超时之前有其它任务或中断服务程序往这个队列里面写入了数据,此时这些任务里面优先级最高的任务将转换为就绪态来准备读取队列里面的数据,如果这里多个任务的优先级相同,那么等待时间最久的任务将优先转换为就绪态来准备读取队列里面的数据。
      F r e e R T O S FreeRTOS FreeRTOS里面的写队列操作也支持一个可选的 阻塞时间参数,具体说明和读操作类似,这里就不在累述了。这里还需要注意的是 F r e e R T O S FreeRTOS FreeRTOS中的队列操作,比如读,写,当然还有其它操作,的接口都有基本版本以及中断版本,中断版本的接口都带有 I S R ISR ISR后缀,中断版本没有阻塞时间参数,中断版本的接口只能在中断服务程序中调用
     其实队列还是比较简单的,有了以上基础再看看图7的文档中第4章 F r e e R T O S FreeRTOS FreeRTOS官方对队列管理的介绍以及图8中第3章队列管理的相应的接口介绍就可以开始实际写代码操作了。

图7.
图8.

     其实队列还是比较简单的,有了以上基础再看看图7的文档中第4章 F r e e R T O S FreeRTOS FreeRTOS官方对队列管理的介绍以及图8中第3章队列管理的相应的接口介绍就可以开始实际写代码操作了。下面我们来看一个简单的例子,这个例子中我们定义了三个任务一个队列,队列里面有3个数据项,其中两个任务向队列里面发送数据,一个任务从队列里面读取数据,两个发送数据的任务的优先级相同且大于接收数据的任务的优先级。因为两个发送数据的任务(阻塞时间设置为100 m s ms ms)的优先级大于数据接收任务的优先级,因此轮到接收数据的任务接收数据的时候队列应该已经满了,所以接收任务的阻塞时间设置为0。对于数据发送任务只要数据接收任务读取了队列之中的一个数据项,阻塞的数据发送任务马上就会抢占数据接收任务来进行数据的发送直到队列再次满而进入阻塞状态,这时数据接收任务就开始了数据的接收。
     该例子的主要代码都在下面了,这里队列里面存储的数据项是一个包含两个元素的结构体一个用来区别发送的数据到底是来自于两个发送任务中的哪一个,另一个用来存储实际的数据。这里一个任务发送数据100,另一个任务发送数据200。

static QueueHandle_t xQueue=NULL;	/* Define an enumerated type used to identify the source of the data. */
typedef enum
{Task1Sender,Task2Sender
} DataSource_t;
/* Define the structure type that will be passed on the queue. */
typedef struct
{uint8_t ucValue;DataSource_t eDataSource;
} Data_t;/* Declare two variables of type Data_t that will be passed on the queue. */
static const Data_t xStructsToSend[ 2 ] =
{{ 100, Task1Sender }, /* Used by Task 1. */{ 200, Task2Sender }  /* Used by Task 2. */
};static void vSenderTask( void *pvParameters )
{BaseType_t xStatus;const TickType_t xTicksToWait = pdMS_TO_TICKS( 100 );/* As per most tasks, this task is implemented within an infinite loop. */while(1){/* Send to the queue.The second parameter is the address of the structure being sent. Theaddress is passed in as the task parameter so pvParameters is useddirectly.The third parameter is the Block time - the time the task should be keptin the Blocked state to wait for space to become available on the queueif the queue is already full. A block time is specified because thesending tasks have a higher priority than the receiving task so the queueis expected to become full. The receiving task will remove items fromthe queue when both sending tasks are in the Blocked state. */xStatus = xQueueSendToBack( xQueue, pvParameters, xTicksToWait );if( xStatus != pdPASS ){/* The send operation could not complete, even after waiting for 100ms.This must be an error as the receiving task should make space in thequeue as soon as both sending tasks are in the Blocked state. */printf( "Could not send to the queue.\r\n" );}}
}static void vReceiverTask( void *pvParameters )
{/* Declare the structure that will hold the values received from the queue. */Data_t xReceivedStructure;BaseType_t xStatus;/* As per most tasks, this task is implemented within an infinite loop. */while(1){/* Because it has the lowest priority this task will only run when thesending tasks are in the Blocked state. The sending tasks will only enterthe Blocked state when the queue is full so this task always expects thenumber of items in the queue to be equal to the queue length, which is 3 inthis case. */if( uxQueueMessagesWaiting( xQueue ) != 3 ){printf( "Queue should have been full!\r\n" );}/* Receive from the queue.The second parameter is the buffer into which the received data will beplaced. In this case the buffer is simply the address of a variable thathas the required size to hold the received structure.The last parameter is the block time - the maximum amount of time that thetask will remain in the Blocked state to wait for data to be availableif the queue is already empty. In this case a block time is not necessarybecause this task will only run when the queue is full. */xStatus = xQueueReceive( xQueue, &xReceivedStructure, 0 );if( xStatus == pdPASS ){/* Data was successfully received from the queue, print out the receivedvalue and the source of the value. */if( xReceivedStructure.eDataSource == Task1Sender ){printf( "From Sender 1 = %d\r\n", xReceivedStructure.ucValue );}else{printf( "From Sender 2 = %d\r\n", xReceivedStructure.ucValue );}}else{/* Nothing was received from the queue. This must be an error as thistask should only run when the queue is full. */printf( "Could not receive from the queue.\r\n" );}}
}int main(void)
{	 NVIC_PriorityGroupConfig( NVIC_PriorityGroup_4);	uart_init(115200);	/* The queue is created to hold a maximum of 3 structures of type Data_t. */xQueue = xQueueCreate( 3, sizeof( Data_t ) );if( xQueue != NULL ){/* Create two instances of the task that will write to the queue. Theparameter is used to pass the structure that the task will write to thequeue, so one task will continuously send xStructsToSend[ 0 ] to the queuewhile the other task will continuously send xStructsToSend[ 1 ]. Bothtasks are created at priority 2, which is above the priority of the receiver. */xTaskCreate( vSenderTask, "Sender1", 1000, (void *) &( xStructsToSend[ 0 ] ), 2, NULL );xTaskCreate( vSenderTask, "Sender2", 1000, (void *) &( xStructsToSend[ 1 ] ), 2, NULL );/* Create the task that will read from the queue. The task is created withpriority 1, so below the priority of the sender tasks. */xTaskCreate( vReceiverTask, "Receiver", 1000, NULL, 1, NULL );/* Start the scheduler so the created tasks start executing. */vTaskStartScheduler();}else{/* The queue could not be created. */}/* If all is well then main() will never reach here as the scheduler willnow be running the tasks. If main() does reach here then it is likely thatthere was insufficient heap memory available for the idle task to be created.Chapter 2 provides more information on heap memory management. */while(1);   	
}

该例子的工程代码在这里。

图1.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/642589.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Adobe Media Encoder 2023下载安装教程,ME 2023安装教程,附安装包和工具,无套路,轻松搞的安装

前言 Adobe Media Encoder是一个视频和音频编码应用程序,可让针对不同应用程序和观众,以各种分发格式对音频和视频文件进行编码。包括专门设计的预设设置,以便导出与特定交付媒体兼容的文件,可以按适合多种设备的格式导出视频&am…

漫漫数学之旅010

文章目录 经典格言数学习题古今评注科学家小传(一)艾伦凯(二)托马斯C黑尔斯 经典格言 计算机的归宿是融入我们的生活,就像其它一切我们习以为常的东西:手表、纸、铅笔和衣服,我们不再把它们看作…

Hbas简介:数据模型和概念、物理视图

文章目录 说明零 BigTable一 Hbase简介二 HBase 访问接口简介三 行式&列式存储四 HBase 数据模型4.1 HBase 列族数据模型4.2 数据模型的相关概念4.3 数据坐标 五 概念&物理视图 说明 本文参考自林子雨老师的大数据技术原理与应用(第三版)教材内容,仅供学习…

maven pom中的内置变量及引用

目录 前言内置变量引用 前言 maven其实有很多内置变量供开发着在开发中使用,比如说basedir这变量,它指的是pom.xml文件所在的目录,下面我们一起来认识一下。 内置变量 变量名作用说明basedir 、project.basedir项目的根目录即包含 pom.xml 文…

Gen AI大潮来袭!8个Salesforce新岗位,你会选择哪个?

人工智能席卷全球,企业对如何整合GenAI有着浓厚的兴趣。为启动企业的GenAI转型浪潮,Salesforce宣布与埃森哲和德勤建立合作伙伴关系,并计划推出更多支持项目。 目前,Salesforce领域的其他咨询公司正在提高员工技能,以…

C++之类型定义

# define QT_PREPEND_NAMESPACE(name) ::name template <class T> struct QIntegerForSizeof: QIntegerForSize<sizeof(T)> { }; typedef QIntegerForSizeof<void*>::Unsigned quintptr; typedef QT_PREPEND_NAMESPACE(quintptr) WId; WId winid; ​ 这段代…

笨蛋学设计模式行为型模式-状态模式【20】

行为型模式-状态模式 8.7状态模式8.7.1概念8.7.2场景8.7.3优势 / 劣势8.7.4状态模式可分为8.7.5状态模式8.7.6实战8.7.6.1题目描述8.7.6.2输入描述8.7.6.3输出描述8.7.6.4代码 8.7.7总结 8.7状态模式 8.7.1概念 ​ 状态模式是指对象在运行时可以根据内部状态的不同而改变它们…

Apache Zeppelin结合Apache Airflow使用1

Apache Zeppelin结合Apache Airflow使用1 文章目录 Apache Zeppelin结合Apache Airflow使用1前言一、安装Airflow二、使用步骤1.目标2.编写DAG2.加载、执行DAG 总结 前言 之前学了Zeppelin的使用&#xff0c;今天开始结合Airflow串任务。 Apache Airflow和Apache Zeppelin是两…

C语言数据结构(3)——线性表其二(单链表)

欢迎来到博主的专栏——C语言数据结构 博主id&#xff1a;代码小豪 文章目录 单链表不连续存储的线性表单链表单链表的结构头指针单链表的操作打印单链表 空链表单链表的插入尾插法 头插法 单链表的查找任意位置处的节点插入单链表节点的删除 销毁链表 单链表 顺序表是一个物…

万字长文详解Java线程池面试题

王有志&#xff0c;一个分享硬核 Java 技术的互金摸鱼侠 加入 Java 人的提桶跑路群&#xff1a;共同富裕的Java人 今天是《面霸的自我修养》第 6 篇文章&#xff0c;我们一起来看看面试中会问到哪些关于线程池的问题吧。数据来源&#xff1a; 大部分来自于各机构&#xff08;J…

【K8S】Kubernetes 中滚动发布由浅入深实战

目录 一、Kubernetes中滚动发布的需求背景1.1 滚动发布1.2 滚动发布、蓝绿发布、金丝雀发布的区别 二、Kubernetes中实现滚动发布2.1 定义Kubernetes中的版本2.2 创建 Deployment 资源对象2.2.1 在 Yaml 中定义 Deployment 资源对象2.2.2 执行命令创建 Deployment 资源对象 三、…

Asp.net core 框架入门

概述 appsettings.json&#xff1a;配置文件&#xff0c;数据库连接字符串配置信息 Program.cs&#xff1a;程序入口文件&#xff08;里面有个Main方法&#xff09; Startup.cs&#xff1a;启动配置文件 依赖项&#xff1a;管理项目所依赖的第三方组件的安装&#xff0c;配…

WampServer

开发笔记 推荐链接php无法保存SESSION问题部署SSL时候产生的问题 推荐链接 链接目录 php无法保存SESSION问题 php.ini文件和phpForApache.ini 文件 里面都有 对路径的控制&#xff0c;相关路径问题可能也需要进行修改&#xff0c;打开文件搜索wamp64或wamp 就可以看到了&…

“深入理解RabbitMQ交换机的原理与应用“

深入理解RabbitMQ交换机的原理与应用 引言1. RabbitMQ交换机简介介绍1.1 什么是RabbitMQ&#xff1f;1.1.1 消息中间件的作用1.1.2 RabbitMQ的特点和优势 1.2 RabbitMQ的基本概念1.2.1 队列1.2.2 交换机1.2.3 路由键 1.3 交换机的作用和分类1.3.1 直连交换机&#xff08;direct…

leetcode 热题100-学习计划

一、题目链接 1. 两数之和 - 力扣&#xff08;LeetCode&#xff09; 解题思路 暴力破解——解题代码 class Solution:def twoSum(self, nums: List[int], target: int) -> List[int]:n len(nums)result []for i in range(0,n):for j in range(i 1,n):#print(i,j)sum …

VS Code Json格式化插件-JSON formatter

&#x1f9aa;整个文件格式化 按快捷键Shift Alt F &#x1f96a;仅格式化选择内容 需要选择完整的json段落即&#xff1a;{} 或 [] 括起来的部分&#xff0c;再按快捷键Ctrl K F

社区公益培训系统功能说明

社区公益培训系统功能说明 本系统将用于社区面向居民开展的公益培训课程展示&#xff0c;在线报名&#xff0c;并按班级排课上课&#xff0c;上课时学员要扫码签到&#xff0c;经常旷课的学员将禁止再报名其他课程。 1. 用户注册与登录 - 提供用户注册和登录功能&#xff0c;…

鸿蒙不再兼容安卓,鸿蒙开发薪资高达4w+,程序员是否需转行鸿蒙?

鸿蒙系统的崛起 鸿蒙系统的推出经历了长时间的研发和完善&#xff0c;它是一款自主研发的操作系统&#xff0c;集成了最新的技术和创新理念。该系统具备卓越的安全性、兼容性和扩展性&#xff0c;因此备受关注。最初&#xff0c;鸿蒙系统主要应用于华为手机产品&#xff0c;但…

惬意上手Python —— 装饰器和内置函数

1. Python装饰器 Python中的装饰器是一种特殊类型的函数&#xff0c;它允许用户在不修改原函数代码的情况下&#xff0c;增加或修改函数的行为。 具体来说,装饰器的工作原理基于Python的函数也是对象这一事实&#xff0c;可以被赋值给变量、作为参数传递给其他函数或者作为其他…

比较有创意的网站

有创意的网站通常展示了独特的设计、交互或内容。以下是一些备受赞誉的有创意的网站&#xff0c;你可以参考&#xff1a; Awwwards: Awwwards 是一个评选并展示全球最优秀网站的平台。你可以在这里找到很多有创意的网站设计。 Awwwards CSS Design Awards: 类似于Awwwards&…