6.FreeRTOS_队列进行任务间通信

本文内容的基础知识点在博文 “ FreeRTOS_队列基础知识 ”中,博文链接如下:

FreeRTOS_队列基础知识-CSDN博客

本文的代码基础在博文  “ FreeRTOS_任务创建与删除 ”中,博文链接如下:

FreeRTOS_任务创建与删除-CSDN博客

队列实现任务间通信

队列实现任务间通信的步骤有3步:

  • 定义一个队列句柄
  • 创建队列
  • 在两个任务中分别写队列和读队列

具体代码如下:

/* 任务1处理函数 */
void Task1Function(void *param){
	
	int i=0;
	while(1){
		
		xQueueSend((QueueHandle_t)param,&i,portMAX_DELAY);/* 写队列 */
		printf("i%d ",i);
		i++;
	}
}
/* 任务2处理函数 */
void Task2Function(void *param){
	
	int j=0;
	while(1){
		xQueueReceive((QueueHandle_t)param,&j,portMAX_DELAY);/* 读队列 */
		printf("j%d ",j);
	}
}

int main( void )
{
#ifdef DEBUG
  debug();
#endif
	
	TaskHandle_t xHandleTask1;
	TaskHandle_t xHandleTask2;
	QueueHandle_t QueueHandle_Test;/* 定义一个队列句柄,这是个指针 */
	
	
	prvSetupHardware();
	SerialPortInit();
	printf("UART TEST\r\n");
	
	QueueHandle_Test = xQueueCreate(2,sizeof(int));/* 创建队列 */
	if(QueueHandle_Test == NULL){
		printf("queue create fail\r\n");
	} 
	
	xTaskCreate(Task1Function,"Task1",100,(void*)QueueHandle_Test,1,&xHandleTask1);
	xTaskCreate(Task2Function,"Task2",100,(void*)QueueHandle_Test,1,&xHandleTask2);
	
	vTaskStartScheduler();
	
	return 0;
}

代码注意点:

当调用xQueueSend传入 &i 时,并不是说可以在队列中可以改变 i 的值,而是利用 i 的地址进行拷贝,即:将 i 的值复制到队列空间中。

运行结果如下:

从结果的红色线和黄色线中可以看到,Task1发送出去的 i 值已经被Task2接收到并存入了 j 变量中,这就说明了队列实现了任务间的通信。

这里紫色框发生了个问题,j5打印之后并没有打印空格符号就直接打印了i6,这说明Task2在运行printf("j%d ",j)这一行时,打印到%d后,就被调度器切换出去了,从而打印了Task1中的内容。可以看到i7之后有两个空格,这是因为Task1在运行printf("i%d ",i)这一行后,切换到了Task2,Task2按照之前未完成的工作继续执行,即:继续打印printf("j%d ",j)中的空格符。

对于紫色框的问题,使用互斥量来保护printf,即可解决。

队列传输大数据的方法

由于调用xQueueSend传输数据时,本质是将传入地址处的数据进行拷贝,这中方法对于很大的数据进行传输时,效率就变得很低。

因此,在传输大数据时,将数据的地址进行传入,之后读取时读取地址,再通过地址访问数据。这样每次写入队列的数据大小就是4字节(32位机),大大提高了传输效率。

在代码中,需要注意的点如下:

  • 创建队列时,每一个数据的大小为sizeof(void*),代表每一个项是一个指针
  • 写入队列时,需定义一个void*类型的指针指向首地址,并把指针的地址传入队列进行复制
  • 读取队列时,读取后指针类型为void*,需要强转成所需类型的指针后使用

具体代码实现如下:

void Task1Function(void *param){
	
	char large_buf[] = "abcdefgasdasdaasdasdasdasdasdasdasdasdhi";/* 大数据 */

	void* p = (void*)large_buf;/* 定义一个void*类型的指针,保存大数据的首地址 */

	while(1){
        /* 写入&p,从p地址中拷贝数据,这个数据就是大数据的首地址 */
		xQueueSend((QueueHandle_t)param,&p,portMAX_DELAY);
		vTaskDelay(100);
	}
}
void Task2Function(void *param){
	
	void* p=NULL;/* 定义一个void*类型的指针,保存大数据的首地址 */
	while(1){

        /* 写入&p,将数据拷贝到p地址中,这个数据就是大数据的首地址 */
		xQueueReceive((QueueHandle_t)param,&p,portMAX_DELAY);

        /* 打印时,要强转为char* */
		printf("p2 = %s\r\n",(char*)p);
	
	}
}

int main( void )
{
#ifdef DEBUG
  debug();
#endif
	
	TaskHandle_t xHandleTask1;
	TaskHandle_t xHandleTask2;
	QueueHandle_t QueueHandle_Test;
	
	prvSetupHardware();
	SerialPortInit();
	printf("UART TEST\r\n");
	
    /* 注意:这里大小为sizeof(void*),代表是一个指针 */
	QueueHandle_Test = xQueueCreate(1,sizeof(void*));
	if(QueueHandle_Test == NULL){
		printf("queue create fail\r\n");
	}
	
	xTaskCreate(Task1Function,"Task1",100,(void*)QueueHandle_Test,2,&xHandleTask1);
	xTaskCreate(Task2Function,"Task2",100,(void*)QueueHandle_Test,1,&xHandleTask2);
	//xTaskCreateStatic(Task3Function,"Task3",100,NULL,1,xTask3Stack,&xTask3TCB);
	vTaskStartScheduler();
	
	return 0;
}

运行结果如下:

  • 9
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值