FreeRTOS入门(03):队列、信号量、互斥量

news2025/7/28 7:51:32

文章目录

  • 目的
  • 队列(queue)
  • 信号量(semaphore)
  • 互斥量(mutex)
    • 互斥量
    • 递归互斥量
  • 总结

目的

FreeRTOS提供给用户最核心的功能是任务(Task),实际项目中通常会有多个任务,任务间多数时候会需要配合工作,这时候就需要用到 队列、信号量、互斥量 等功能了,这篇文章将对相关内容做个介绍。

本文接上篇:《FreeRTOS入门(02):任务基础使用与说明》

队列(queue)

队列是用于任务间传递数据最常用的东西。队列相当于一个缓存,可以向里面写数据,也可以从里面拿数据,遵循先进先出的原则(FIFO),先写入的数据在取的时候也会先取出。

向队列中写数据则队列的可用空间将减少,从队列中读取数据后该数据将从队列中删除,队列的可用空间会增加。如果队列已满的情况下向其中写数据将阻塞直到有空间可以写入;如果从队列中读取时队列为空将阻塞直到有数据可以读取。

读写操作时如果有多个任务都在等待,那么优先级最高的任务将先获得执行权限,如果优先级相同,那么等待时间最久的任务获得执行权限。

队列操作常用的一些函数如下:

// 创建队列,成功的话返回队列句柄,失败则返回NULL
// uxQueueLength为队列长度,uxItemSize为每个数据的容量(单位为字节)
QueueHandle_t xQueueCreate( UBaseType_t uxQueueLength, UBaseType_t uxItemSize );
// 将队列恢复到空的状态
BaseType_t xQueueReset( QueueHandle_t xQueue );
// 删除队列
void vQueueDelete( QueueHandle_t xQueue )

// 写入数据到队列(队尾)
// pvItemToQueue为数据起始地址指针,会从这里开始读入uxItemSize字节数据到队列
// xTicksToWait为队满无法写入时的写入操作超时时间(以Tick计,可以使用pdMS_TO_TICKS(ms)将毫秒时间转换成Tick)
// INCLUDE_vTaskSuspend为1时,xTicksToWait为portMAX_DELAY时将无限期阻塞(不会超时)
// 写入成功返回pdTRUE,否则返回errQUEUE_FULL
BaseType_t xQueueSend( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );
// 功能同上
BaseType_t xQueueSendToBack( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );

// 写入数据到队列(队尾)用于中断中使用,不会阻塞
// pxHigherPriorityTaskWoken可写NULL
BaseType_t xQueueSendFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);
// 功能同上
BaseType_t xQueueSendToBackFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);

// 写入数据到队列(队首)
// 写数据到队首其实不怎么符合队列本身设计思想的,不过特殊情况下还是有这个需求的
BaseType_t xQueueSendToFront( QueueHandle_t xQueue, const void * pvItemToQueue, TickType_t xTicksToWait );
// 写入数据到队列(队首)用于中断中使用,不会阻塞
BaseType_t xQueueSendToFrontFromISR( QueueHandle_t xQueue, const void *pvItemToQueue, BaseType_t *pxHigherPriorityTaskWoken);

// 从队列读取uxItemSize字节数据到pvBuffer,读取完成后队首指针将移动到下一个数据
// 成功返回pdTRUE,否则返回pdFALSE(超时)
BaseType_t xQueueReceive( QueueHandle_t xQueue, void *pvBuffer, TickType_t xTicksToWait );
// 上面函数用于中断中使用的形式,不会阻塞
BaseType_t xQueueReceiveFromISR( QueueHandle_t xQueue, void *pvBuffer, BaseType_t *pxHigherPriorityTaskWoken );

// 获取队首数据,并且队首指针不移动
BaseType_t xQueuePeek( QueueHandle_t xQueue, void *pvBuffer, TickType_t xTicksToWait );
// 上面函数用于中断中使用的形式,不会阻塞
BaseType_t xQueuePeekFromISR( QueueHandle_t xQueue, void *pvBuffer );

除了上面列出的一些操作队列的函数,还有一些查询队列以写、可写、队空、队满等功能的函数,如果有需要可以查询官方文档。

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "queue.h"     // 引入头文件

QueueHandle_t xQueue; // 队列句柄

void task1(void *pvParameters) {
    while(1)
    {
        int tick = xTaskGetTickCount();
        xQueueSend(xQueue, &tick, portMAX_DELAY); // 向队列写数据
        vTaskDelay(500);
    }
}

void task2(void *pvParameters) {
    while(1)
    {
        int data;
        xQueueReceive(xQueue, &data, portMAX_DELAY); // 从队列读取数据
        printf("Task2 data is %d\r\n", data);
    }
}

int main(void) {
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    SystemCoreClockUpdate();
    Delay_Init();
    USART_Printf_Init(115200);

    xQueue = xQueueCreate(3, 4); // 创建队列,队列长度为3,每个空间容纳4个字节

    xTaskCreate(task1, "task1", 256, NULL, 5, NULL); // 创建一个任务
    xTaskCreate(task2, "task2", 256, NULL, 5, NULL); // 创建一个任务

    vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环

    while(1) {} // 程序不会运行到这里
}

在这里插入图片描述

信号量(semaphore)

队列用于任务间传递数据使用,使用时会占据比较大的空间,而且读写性能稍差。有些时候任务间交互可能并不需要传递数据,只要知道 资源 有还是没有、或者有多少而已,这个时候使用信号量就比较好了。

信号量分为二进制信号量和计数信号量两种。二进制只有两个状态,表示资源有或者没有(也叫做二值信号量);计数信号量可以表达出资源的数量。信号量的值比较常见的是0、1、2、3……,0表示没有资源可用。

写信号量时,如果不可写入也不会阻塞,会直接返回错误码,这是和队列最大的不同。读信号时如果没有资源则会阻塞。

信号量相关操作函数如下:

// 创建二进制信号量
SemaphoreHandle_t xSemaphoreCreateBinary( void )
// 创建计数信号量
// uxMaxCount为最大计数值,uxInitialCount为初始计数值
SemaphoreHandle_t xSemaphoreCreateCounting( UBaseType_t uxMaxCount, UBaseType_t uxInitialCount )

// 删除信号量
void vSemaphoreDelete( SemaphoreHandle_t xSemaphore )

// 给出资源,信号量的值+1,不会阻塞
// 操作成功返回pdTRUE,否则返回errQUEUE_FULL
xSemaphoreGive( SemaphoreHandle_t xSemaphore );
// 上面函数的中断函数中使用的版本
xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore, signed BaseType_t *pxHigherPriorityTaskWoken )

// 获取资源,如果没有资源可用(值为0)则会阻塞知道可用,获取后信号量的值-1
// INCLUDE_vTaskSuspend为1时,xTicksToWait为portMAX_DELAY时将无限期阻塞(不会超时)
// 成功返回pdTRUE,否则返回pdFALSE(超时)
xSemaphoreTake( SemaphoreHandle_t xSemaphore, TickType_t xTicksToWait )
// 上面函数的中断函数中使用的版本
xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore, signed BaseType_t *pxHigherPriorityTaskWoken )

// 返回信号量计数值
UBaseType_t uxSemaphoreGetCount( SemaphoreHandle_t xSemaphore )

下面是信号量使用演示:

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "semphr.h"     // 引入头文件

SemaphoreHandle_t xSemaphore; // 信号量句柄

void task1(void *pvParameters) {
    while(1)
    {
        // xSemaphore初始为1,最大为3
        xSemaphoreGive(xSemaphore); // 给出  +1 = 2
        xSemaphoreGive(xSemaphore); // 给出  +1 = 3
        xSemaphoreGive(xSemaphore); // 给出  此处操作将失败
        xSemaphoreGive(xSemaphore); // 给出  此处操作将失败
        vTaskDelete(NULL);
    }
}

void task2(void *pvParameters) {
    while(1)
    {
        vTaskDelay(500);
        xSemaphoreTake(xSemaphore, portMAX_DELAY); // 获取
        printf("Tick: %d\r\n", xTaskGetTickCount()); // 此例中这行最终会打印三次
    }
}

/* 主函数 */
int main(void) {
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    SystemCoreClockUpdate();
    Delay_Init();
    USART_Printf_Init(115200);

    xSemaphore = xSemaphoreCreateCounting(3, 1); // 创建信号量,最大值3,初始值1

    xTaskCreate(task1, "task1", 256, NULL, 5, NULL); // 创建一个任务
    xTaskCreate(task2, "task2", 256, NULL, 5, NULL); // 创建一个任务

    vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环

    while(1) {}
    // 程序不会运行到这里
}

在这里插入图片描述

互斥量(mutex)

互斥量其实也是一种信号量,特别是普通的互斥量其实和二进制信号量挺像的。需要注意的是 中断函数中不能使用互斥量

互斥量

普通的互斥量,和二进制信号量很像,其它操作函数基本都是通用的(不能在中断中使用)。创建互斥量函数如下:

SemaphoreHandle_t xSemaphoreCreateMutex( void )

互斥量注意是用来对特定的资源(比如全局变量、外设等)进行保护用的。比如某个任务使用串口输出信息时,如果有优先级更高的任务插入进来,也使用这个串口输出信息,那可能最终串口输出的内容可能就会穿插糅合再一起了。这通常是不符合要求的,所以就需要用互斥量了。

互斥量也被叫做互斥锁,某个任务要用资源时先申请互斥量上个锁,用完资源后解锁。锁在被锁上的时候其它任务无法重复上锁,会阻塞到锁被释放后才能成功上锁。使用互斥量时需要注意的是上锁和解锁要成对出现,并且谁上锁谁就必须解锁。

下面是个互斥量使用演示:

#include "debug.h"
#include "FreeRTOS.h" // 引入头文件
#include "task.h"     // 引入头文件
#include "semphr.h"     // 引入头文件

SemaphoreHandle_t xMutex; // 互斥量句柄

void task(void *pvParameters) {
    while(1)
    {
        xSemaphoreTake(xMutex, portMAX_DELAY); // 上锁
        printf((const char *)pvParameters);
        xSemaphoreGive(xMutex);  // 解锁
        vTaskDelay(5);
    }
}

/* 主函数 */
int main(void) {
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    SystemCoreClockUpdate();
    Delay_Init();
    USART_Printf_Init(9600); // 9600波特率下每秒约可发送960字节

    xMutex = xSemaphoreCreateMutex(); // 创建互斥量

    xTaskCreate(task, "task1", 256, "USART1_Task1: ABCDEFGHIJKLMNOPQRSTUVWXYZ\r\n", 5, NULL); // 创建一个任务
    xTaskCreate(task, "task2", 256, "USART1_Task2: 01234567890123456789\r\n", 7, NULL); // 创建一个任务 该任务优先级更高,会抢占前面的任务

    vTaskStartScheduler(); // 任务调度,任务将在这里根据情况开始运行,程序将在这里无序循环

    while(1) {} // 程序不会运行到这里
}

在这里插入图片描述
上面演示中 task2 有更高的优先级,会抢占 task1 ,所以 task2 有更多的机会运行,也就有更多时间可以使用串口输出信息。但是因为有互斥量存在,所以每个任务在上锁之后使用串口,在解锁前都可以独占串口,不会被抢占,所以最终都能完整的输出信息。

如果注释掉代码中上锁和解锁操作,那么最终输出会变成下面这样:
在这里插入图片描述
仔细看可以发现,因为 task2 有更高优先级,不会被 task1 抢占,所以 task2 可以完整的输出信息。但是 task1 输出过程中就会被 task2 插入打断。

递归互斥量

上面的互斥量有点像二进制信号量,而递归互斥量就有点像计数信号量。递归互斥量可以给多次上锁,但用完后上了几次锁就需要解几次锁。

递归互斥量部分函数也可以和上面公用,但下面几个是它特有的:

SemaphoreHandle_t xSemaphoreCreateRecursiveMutex( void )
xSemaphoreGiveRecursive( SemaphoreHandle_t xMutex )
xSemaphoreTakeRecursive( SemaphoreHandle_t xMutex, TickType_t xTicksToWait );

总结

队列、信号量、互斥量的使用比较简单,这几个也是大多数RTOS都有的功能。

FreeRTOS中用于任务之间配合工作的附加功能除了这几个还有其它一些,比如任务通知、事件组、队列集、流与消息缓冲区等。这些功能可以为项目开发带来更多的便利性,当然没有它们项目需求也能实现。有时间的话会在后续的文章中介绍这些功能。

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

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

相关文章

【GO】K8s 管理系统项目9[API部分--Secret]

K8s 管理系统项目[API部分–Secret] 1. 接口实现 service/dataselector.go // secret type secretCell corev1.Secretfunc (s secretCell) GetCreation() time.Time {return s.CreationTimestamp.Time }func (s secretCell) GetName() string {return s.Name }2. Secret功能…

浅谈BOM

什么是BOM BOM对于每个前端都不陌生,但是很多人都停留在表面,而没有深层次的研究过它。JavaScript有一个非常重要的运行环境就是浏览器,而且浏览器本身又作为一个应用程序需要对其本身进行操作,所以通常浏览器会有对应的对象模型…

tidb ptca,ptcp考证

PingCAP 认证 TiDB 数据库专员 V6 考试(2023-02-23)https://learn.pingcap.com/learner/exam-market/list?categoryPCTA PingCAP 认证 TiDB 数据库管理专家(PCTP - DBA)认证考试范围指引 - ☄️ 学习与认证 - TiDB 的问答社区:lo…

Linux部署的Java应用生成图片和二维码会出现中文乱码的解决办法

Linux部署的Java应用生成图片和二维码会出现中文乱码,这是因为没有中文字体的原因,需要安装字体库。下载字体库https://download.csdn.net/download/a506602491/87490755,将文件解压至 /usr/share/fonts 目录下,如果没有fonts文件…

TRichView改进对HTML的支持并增强报告功能

TRichView改进对HTML的支持并增强报告功能 TRichView v21.0新功能: HTML导入和导出改进。 加载和保存的新方法。 改进图像选择和插入。 RichView Actions v11.0新功能: 包括加载和保存HTML的新方法。 使用新的TRVAControlPanel.OnChoosePicture事件选择…

[数据结构]链表OJ

目录 数据结构之链表OJ:: 1.移除链表元素 2.反转链表 3.链表的中间结点 4.链表中倒数第k个结点 5.合并两个有序链表 6.链表分割 7.链表的回文结构 8.相交链表 9.环形链表 10.环形链表II 11.复制带随机指针的链表 数据结构之链表OJ:&#xff…

springboot自定义starter时使用@AutoConfigureBefore、@AutoConfigureAfter的细节问题

正常利用springboot的自动装配 ConfB Configuration(proxyBeanMethodsfalse) public class ConfB {public ConfB(){System.out.println("ConfB构造方式执行...");} }不加spring.factories 项目包结构 此时resources中没有spring.factories 执行结果 2023-02-24…

运动蓝牙耳机什么牌子好,运动蓝牙耳机品牌推荐

现在市面上运动耳机的品牌越来越多,还不知道选择哪一些运动耳机品牌,可以看看下面的一些耳机分享,运动耳机需要注意耳机的参数配置以及佩戴舒适度,根据自己最根本的使用需求来选择运动耳机。 1、南卡Runner Pro4骨传导蓝牙运动耳…

剑指 Offer 46. 把数字翻译成字符串

剑指 Offer 46. 把数字翻译成字符串 难度:middle\color{orange}{middle}middle 其实就是有条件的 青蛙跳格子 问题。 题目描述 给定一个数字,我们按照如下规则把它翻译为字符串:0 翻译成 “a” ,1 翻译成 “b”,………

GitHub狂飙30K+star面试现场,专为程序员面试打造,现已开源可下载

《程序员面试现场》上线2个月已经在GitHub上已经狂飙到30Kstar(能在在GitHub上拿到30K的star,有没有干货,我就不多说了)总结的很全面,主要是针对面试之前的准备工作,分为知彼、知己、问答、贯通、综合五部分…

06- OpenCV查找图像轮廓 (OpenCV系列) (机器视觉)

知识重点 灰度图转换: gray cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)二值化: 返回两个东西,一个阈值, 一个是二值化的图: thresh, binary cv2.threshold(gray, 150, 255, cv2.THRESH_BINARY)查找轮廓: 返回两个结果,分别是轮廓和层级: c…

白帽黑客入行应该怎么学?零基础小白也能轻松上手!

这几年随着我国《国家网络空间安全战略》《网络安全法》《网络安全等级保护2.0》等一系列政策/法规/标准的持续落地,网络安全行业地位、薪资随之水涨船高。 1为什么网络安全行业是IT行业最后的红利? 根据腾讯安全发布的《互联网安全报告》,…

Python每日一练(20230224)

目录 1. 列表奇偶拆分 ★ 2. 二叉树的后序遍历 ★★ 3. 接雨水 ★★★ 附录 二叉树 特点 性质 特殊二叉树 满二叉树 完全二叉树 完全二叉树性质 二叉树的遍历 1. 列表奇偶拆分 【问题描述】 输入一个列表,包含若干个整数(允许为空&#xff…

Spring Batch 高级篇-并行步骤

目录 引言 概念 案例 转视频版 引言 接着上篇:Spring Batch 高级篇-多线程步骤,了解Spring Batch多线程步骤后,接下来一起学习一下Spring Batch 高级功能-并行步骤 概念 并行步骤,指的是某2个或者多个步骤同时执行。比如下…

Ask林曦|来回答,30个你关心的日常问题(二)

在林曦老师的线上书法直播课上,上课前后的聊天时间里,时常有同学向林曦老师提问,这些问题涵盖了日常生活的诸多方面,从身体的保养,到快乐的法门,皆是大家感兴趣的,也都共同关切的。   暄桐教室…

python+Vue学生作业系统 django课程在线学习网站系统

系统分为学生,教师,管理员三个角色: 学生功能: 1.学生注册登录系统 2.学生查看个人信息,修改个人信息 3.学生查看主页综合评价,查看今日值班信息 4.学生在线申请请假信息,查看请假的审核结果和请…

180、【动态规划】leetcode ——583. 两个字符串的删除操作:两种动态规划思路(C++版本)

题目描述 原题链接:583. 两个字符串的删除操作 解题思路 (1)基于求最长公共子序列思路 本题与 1143. 最长公共子序列 的区别在于,1143中求的是两个序列中的最长公共子序列,而本题是要找到最少删除多少个元素后可以得…

PHP程序员适合创业吗?

创业是一件自然而然的事,不需要人为选择。 只要你是一个努力能干主动的人,当你在一个行业深耕5年之后,就会发现人生发展的下一步就是创业。当然如果行业合适的话。 什么叫行业合适呢? 就是创业的成本并不那么高,不需…

怎么在LinkedIn领英安全添加到3万个好友?

根据领英最新公布的数据:领英全球用户数已经达到8.3亿,超5800万个公司主页,可以说是世界上最-大的business database。 这就不难理解为什么越来越多的外贸人,开始认真尝试和重视在领英开发客户,因为领英确实是外贸人&a…

鸿蒙3.0 APP混合开发闪退问题笔记

APP采用cordova混合开发, 鸿蒙2.0以及安卓操作系统正常使用,但是在鸿蒙3.0中出现APP闪退,对APP进行真机调试发现,鸿蒙3.0系统对crosswork插件存在兼容问题,这些问题会导致APP页面加载失败,进而导致App闪退测…