首页 > 其他分享 >FreeRTOS入门教程(队列详细使用示例)

FreeRTOS入门教程(队列详细使用示例)

时间:2023-10-07 21:06:24浏览次数:54  
标签:xQueueCalcHandle FreeRTOS 示例 队列 void 入门教程 100 NULL 数据

(文章目录)


前言

上篇文章我们已经讲解了队列的概念和队列相关的API函数,那么本篇文章的话就开始带大家来学习使用队列。

一、队列基本使用

这个例子将会创建三个任务,其中两个任务用来发送数据到队列中,另一个任务用来从队列中读取数据。

void Task1Function(void * param)
{
	int val;
	
	while (1)
	{
		val = 100;
		xQueueSend(xQueueCalcHandle, &val, 0);
		
		vTaskDelay(1000);
	}
}

void Task2Function(void * param)
{
	int val;
	
	while (1)
	{
		val = 200;
		xQueueSend(xQueueCalcHandle, &val, 0);
		
		vTaskDelay(1000);
	}
}

void Task3Function(void * param)
{
	int val;
	const TickType_t xTicksToWait = pdMS_TO_TICKS( 100UL );
	BaseType_t xStatus;
	
	while (1)
	{
		xStatus = xQueueReceive(xQueueCalcHandle, &val, xTicksToWait);
		if( xStatus == pdPASS )
		{
			/* 读到了数据 */
			printf( "Received = %d\r\n", val );
		} 
		else
		{
			/* 没读到数据 */
			printf( "Could not receive from the queue.\r\n" );
		}		
	}
}


xQueueCalcHandle = xQueueCreate(5, sizeof(int));

xTaskCreate(Task1Function, "Task1", 100, NULL, 1, NULL);
xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
xTaskCreate(Task3Function, "Task3", 100, NULL, 2, NULL);


运行效果:

从运行效果中可以看出,当队列中有数据的时候就能够从队列中读取到数据,当队列中没有数据时,超时后会返回pdFALSE。

在这里插入图片描述 这里使用百问网的一张图片来描述这个过程: 在这里插入图片描述

二、如何分辨数据源

通过上面这个实验我们完成了队列数据的发送和队列数据的接收,但是我们无法得知数据是哪个队列所发送的,那么下面这个实验就带大家来完成分辨数据源的实验。

前面的实验中我们使用单独的一个int变量来代表数据,这样的话只能接收到对应的数据而无法分辨是谁发过来的数据,那么有什么办法来分辨是谁发来的数据呢?

这里的解决方法是使用结构体:

typedef enum
{
	Task1,
	Task2
}ID_t;

typedef struct data
{
	ID_t id;
	int data;
}Data_t;

static Data_t senddata[2] = {
	{Task1, 10},
	{Task2, 20}
};

void Task1Function(void * param)
{
	
	while (1)
	{
		xQueueSend(xQueueCalcHandle, &senddata[0], 0);
		
		vTaskDelay(1000);
	}
}

void Task2Function(void * param)
{
	
	while (1)
	{
		xQueueSend(xQueueCalcHandle, &senddata[1], 0);
		
		vTaskDelay(1000);
	}
}

void Task3Function(void * param)
{
	Data_t mydata;
	const TickType_t xTicksToWait = pdMS_TO_TICKS( 100UL );
	BaseType_t xStatus;
	
	while (1)
	{
		xStatus = xQueueReceive(xQueueCalcHandle, &mydata, xTicksToWait);
		if( xStatus == pdPASS )
		{
			/* 读到了数据 */
			if(mydata.id == Task1)
			{
				printf("this is Task1 data :%d\r\n", mydata.data);
			}
			else
			{
				printf("this is Task2 data :%d\r\n", mydata.data);
			}
		} 
		else
		{
			/* 没读到数据 */
			printf( "Could not receive from the queue.\r\n" );
		}		
	}
}


xQueueCalcHandle = xQueueCreate(5, sizeof(Data_t));

xTaskCreate(Task1Function, "Task1", 100, NULL, 1, NULL);
xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
xTaskCreate(Task3Function, "Task3", 100, NULL, 2, NULL);

当接收到数据时会先判断结构体中的id,通过id来判断是哪个任务发送过来的数据。

运行效果: 在这里插入图片描述

三、传输大块数据

FreeRTOS 中的队列通常使用数据拷贝来传递数据。这意味着当你将数据发送到队列或从队列接收数据时,队列会在内部复制数据的副本,而不是传递指向原始数据的指针。

这种数据拷贝的方法确保了数据的安全性和一致性,因为多个任务可以独立访问它们自己的副本,而不会干扰其他任务。然而,需要注意的是,数据拷贝可能会引入一些性能开销,尤其是在处理大量数据时。

那么当使用队列来传输大量数据时该怎么做呢?

这里我们可以使用指针来解决这个问题,传递大块数据的时候我们可以使用指针来解决这个问题,在传输大块数据时,可以先得到数据的地址,将数据的地址作为数据传递过来,当接收到数据的地址时,就能够通过数据的地址来得到对应的数据了。

示例:

char pcbuffer[100] = "Hello World";

void Task1Function(void * param)
{
	char* buffer;
	while (1)
	{
		buffer = pcbuffer;
		xQueueSend(xQueueCalcHandle, &buffer, 0);
		
		vTaskDelay(1000);
	}
}

void Task2Function(void * param)
{
	char* buffer;
	while (1)
	{
		buffer = pcbuffer;
		xQueueSend(xQueueCalcHandle, &buffer, 0);
		
		vTaskDelay(1000);
	}
}

void Task3Function(void * param)
{
	char* rebuffer;
	const TickType_t xTicksToWait = pdMS_TO_TICKS( 100UL );
	BaseType_t xStatus;
	
	while (1)
	{
		xStatus = xQueueReceive(xQueueCalcHandle, &rebuffer, xTicksToWait);
		if( xStatus == pdPASS )
		{
			/* 读到了数据 */
			printf("recv buffer : %s\r\n", rebuffer);
		} 
		else
		{
			/* 没读到数据 */
			printf( "Could not receive from the queue.\r\n" );
		}		
	}
}

xQueueCalcHandle = xQueueCreate(5, sizeof(char*));
if (xQueueCalcHandle == NULL)
{
	printf("can not create queue\r\n");
}

xTaskCreate(Task1Function, "Task1", 100, NULL, 1, NULL);
xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
xTaskCreate(Task3Function, "Task3", 100, NULL, 2, NULL);

运行结果:

在这里插入图片描述

这里有几个点需要注意:

1. 由于队列传递数据是复制数据的副本,所有传输数据时并不会影响到原来的数据,但是在这里使用到了地址,当改变这个地址空间的数据后,原来的数据也会受到影响。

2. 由于传递的是地址空间,那么这里的话就必须保证这个数据是全局数据,因为局部数据会被释放,释放后就无法进行使用了,所有需要保证数据是全局数据。

总结

本篇文章就讲解到这里,本篇文章主要给大家讲解了队列的具体代码和使用方法。

标签:xQueueCalcHandle,FreeRTOS,示例,队列,void,入门教程,100,NULL,数据
From: https://blog.51cto.com/u_16153875/7741982

相关文章

  • golang实现一个简单的文件浏览下载功能代码示例
    想省事用Claude(一个依托chatgpt的AI)生成一段golang的文件浏览下载示例,结果给生成的代码大概是这样的(省去了无关部分,主要部分如下):http.HandleFunc("/*",downloadFile)http.HandleFunc("/",showFileList)测试之后,结果发现每次都会走到“/”下去,无论如何都不会......
  • 【HBuilderX】解决黑色主题中的注释颜色太浅的问题(代码示例)
    "foreground":"#a6a085"打开后,搜索Comment,修改,一个是//的颜色,一个是注释内容的颜色{ "name":"Comment", "scope":"comment", "settings":{ "foreground":"#a6a085" } }, {......
  • 安卓开发组件开发示例
    系统原生下拉刷新<androidx.swiperefreshlayout.widget.SwipeRefreshLayoutandroid:id="@+id/swipe_refresh_layout"android:layout_width="match_parent"android:layout_height="match_parent"xmlns:android="http://sc......
  • 微软写了份GPT-4V说明书:166页详细讲解,提示词demo示例全都有
    克雷西萧箫发自凹非寺量子位公众号QbitAI多模态王炸大模型GPT-4V,166页“说明书”重磅发布!而且还是微软团队出品。什么样的论文,能写出166页?不仅详细测评了GPT-4V在十大任务上的表现,从基础的图像识别、到复杂的逻辑推理都有展示;还传授了一整套......
  • 十九、STM32移植FreeRTOS操作系统
    1.环境安装下载FreeRTOS源码2.移植2.1在STM32项目下创建freertos文件夹,然后在文件夹下创建inc、src、port文件夹2.2将FreeRTOS源码目录下的FreeRTOS/Source/include文件夹下的.h头文件全部复制到STM32项目的freertos/inc文件夹下2.3 将FreeRTOS源码目录下的FreeRTOS/Sourc......
  • Android入门教程 | UI布局之RelativeLayout 相对布局
    RelativeLayout简述RelativeLayout继承于android.widget.ViewGroup,按照子元素之间的位置关系完成布局,作为Android系统五大布局中最灵活也是最常用的一种布局方式,非常适合于一些比较复杂的界面设计。RelativeLayout和LinearLayout类似,都是ViewGroup,能“容纳”多个子view。R......
  • 有关于Mysql的简单问题及示例(增删改查 一对一 多对多 左外连接 右外链接)
    Mysql1、请自行设计表并针对该表练习最基本的增删改查且写出示例代码建立表格class其中有属性nameidgenderinterest表格建立完成向表中插入数据插入数据完成尝试删除表中id=101的数据删除数据成功尝试修改表中id为102的数据修改成功2、请问什么是一对多?请自......
  • R语言代码示例
    以下是一个更为复杂的R语言代码示例,展示了一些复杂的数据处理、可视化和模型构建技巧:#设定随机数种子set.seed(123)#生成模拟数据n<-1000x1<-rnorm(n)x2<-rnorm(n)y<-2*x1+3*x2+rnorm(n)#数据预处理df<-data.frame(x1,x2,y)df$x_sum<-df$x1+df$x......
  • C 语言代码示例
    以下是一个较为复杂的C语言代码示例,它演示了如何使用链表数据结构实现一个简单的图(Graph)数据结构,并实现图的深度优先搜索(DFS)算法:#include<stdio.h>#include<stdlib.h>structNode{intvertex;structNode*next;};structGraph{intnumVertices;st......
  • FreeRTOS入门教程(同步与互斥)
    (文章目录)前言前几篇文章一直在围绕FreeRTOS中的任务创建,删除,优先级,调度算法进行讲解,那么从本篇文章开始将围绕同步与互斥来展开讲解。一、同步与互斥概念当多个任务或线程共享资源并发执行时,同步和互斥是两个关键的概念。1.同步(Synchronization)是指协调多个任务或线程的执......