freeRtos队列的使用

一.队列的特性

队列是一个环形缓冲区

 很多生产者生产很多数据,很多消费者读数据,唤醒之后首先是执行高优先级的,同等优先级先执行等待时间长的。

无论是写数据还是读数据都会有一个超时时间,写数据当队列已经满了就等待一会,当读数据队列已经空了可以等待一会。

二.队列的常规使用

修改之前的代码,之前同步互斥的代码,当任务1执行时,任务2会抢占CPU资源,原本单任务1执行的时间只需要2s,任务2抢占CPU之后执行的时间4s。修改之后任务1计算完之后写入队列中,任务2读取队列,队列中有数据就会读取出来,队列中没有数据就会进入阻塞状态,不会参与CPU调度。

1.创建队列

2.写数据 

 3.读数据

4.任务1和2实现队列同步 

static int sum=0;
static volatile int flagCaleEnd=0;
static volatile int flagUartUsed=0;
static QueueHandle_t xQueueCalHandle;

void Task1Function(void * param)
{
	volatile int i=0;
	while (1)
	{
		for(i=0;i<10000000;i++)
		{
			sum++;
		}
		//flagCaleEnd=1;
		//printf("1");
		//vTaskDelete(NULL);//自杀
		xQueueSend(xQueueCalHandle,&sum,portMAX_DELAY);
		sum=1;
	}
}

void Task2Function(void * param)
{
	int val;
	while (1)
	{
		xQueueReceive(xQueueCalHandle,&val,portMAX_DELAY);
		printf("sum=%d\r\n",val);
	}
}

void TaskGenericFunction(void * param)
{
	while (1)
	{
		if(!flagUartUsed)
		{
			flagUartUsed=1;
			printf("%s\r\n",(char*)param);
			flagUartUsed=0;
			vTaskDelay(1);
		}
	}
}

/*-----------------------------------------------------------*/

int main( void )
{
	TaskHandle_t xHandleTask1;
		
#ifdef DEBUG
  debug();
#endif

	prvSetupHardware();

	printf("Hello, world!\r\n");

	xQueueCalHandle=xQueueCreate(2,sizeof(int)) ;
	if(xQueueCalHandle==NULL)
	{
		printf("can not create queue\r\n");
	}

	xTaskCreate(Task1Function, "Task1", 100, NULL, 1, &xHandleTask1);
	xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
	
	//xTaskCreate(TaskGenericFunction, "Task3", 100, "Task3 is running", 1, NULL);
	//xTaskCreate(TaskGenericFunction, "Task4", 100, "Task4 is running", 1, NULL);

	/* Start the scheduler. */
	vTaskStartScheduler();

	/* Will only get here if there was not enough heap space to create the
	idle task. */
	return 0;
}

 5.任务3和4实现队列互斥

//创建一个队列
//往队列里面写一个数据,如果队列里面有数据代表你可以来读这个队列
//如果某个任务想要执行先GetUARTLock()获得这个锁,读到了代表有使用权
//使用完PutUARTLock()随便写一个数据

//创建一个队列
//往队列里面写一个数据,如果队列里面有数据代表你可以来读这个队列
//如果某个任务想要执行先GetUARTLock()获得这个锁,读到了代表有使用权
//使用完PutUARTLock()随便写一个数据
int InitUARTLock(void)
{
	int val;
	xQueueUARTcHandle=xQueueCreate(1,sizeof(int)) ;
	if(xQueueUARTcHandle==NULL)
	{
		printf("can not create queue\r\n");
		return -1;
	}
	xQueueSend(xQueueCalHandle,&val,portMAX_DELAY);
	return 0;
}

void GetUARTLock(void)
{
	int val;
	xQueueReceive(xQueueCalHandle,&val,portMAX_DELAY);
}

void PutUARTLock(void)
{
	int val;
	xQueueSend(xQueueCalHandle,&val,portMAX_DELAY);
}
static int sum=0;
static volatile int flagCaleEnd=0;
static volatile int flagUartUsed=0;
static QueueHandle_t xQueueCalHandle;
static QueueHandle_t xQueueUARTcHandle;

//创建一个队列
//往队列里面写一个数据,如果队列里面有数据代表你可以来读这个队列
//如果某个任务想要执行先GetUARTLock()获得这个锁,读到了代表有使用权
//使用完PutUARTLock()随便写一个数据
int InitUARTLock(void)
{
	int val;
	xQueueUARTcHandle=xQueueCreate(1,sizeof(int)) ;
	if(xQueueUARTcHandle==NULL)
	{
		printf("can not create queue\r\n");
		return -1;
	}
	xQueueSend(xQueueCalHandle,&val,portMAX_DELAY);
	return 0;
}

void GetUARTLock(void)
{
	int val;
	xQueueReceive(xQueueCalHandle,&val,portMAX_DELAY);
}

void PutUARTLock(void)
{
	int val;
	xQueueSend(xQueueCalHandle,&val,portMAX_DELAY);
}

void Task1Function(void * param)
{
	volatile int i=0;
	while (1)
	{
		for(i=0;i<10000000;i++)
		{
			sum++;
		}
		//flagCaleEnd=1;
		//printf("1");
		//vTaskDelete(NULL);//自杀
		xQueueSend(xQueueCalHandle,&sum,portMAX_DELAY);
		sum=1;
	}
}

void Task2Function(void * param)
{
	int val;
	while (1)
	{
		xQueueReceive(xQueueCalHandle,&val,portMAX_DELAY);
		printf("sum=%d\r\n",val);
	}
}

void TaskGenericFunction(void * param)
{
	while (1)
	{
		GetUARTLock();
		printf("%s\r\n",(char*)param);
					//Task3 is waiting
		PutUARTLock();//Task3 is ready但是Task4还是running所以需要主动休息
		vTaskDelay(1);
	}
}

/*-----------------------------------------------------------*/

int main( void )
{
	TaskHandle_t xHandleTask1;
		
#ifdef DEBUG
  debug();
#endif

	prvSetupHardware();

	printf("Hello, world!\r\n");

	xQueueCalHandle=xQueueCreate(2,sizeof(int)) ;
	if(xQueueCalHandle==NULL)
	{
		printf("can not create queue\r\n");
	}

	InitUARTLock();

	xTaskCreate(Task1Function, "Task1", 100, NULL, 1, &xHandleTask1);
	xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
	
	xTaskCreate(TaskGenericFunction, "Task3", 100, "Task3 is running", 1, NULL);
	xTaskCreate(TaskGenericFunction, "Task4", 100, "Task4 is running", 1, NULL);

	/* Start the scheduler. */
	vTaskStartScheduler();

	/* Will only get here if there was not enough heap space to create the
	idle task. */
	return 0;
}

三.: 邮箱(Mailbox)

四.队列集 

1.创建2个queue

2.创建queue set

3.把2个queue添加进queue set

4.创建3个任务 

static volatile int flagCalcEnd = 0;
static volatile int flagUARTused = 0;
static QueueHandle_t xQueueHandle1;
static QueueHandle_t xQueueHandle2;
static QueueSetHandle_t xQeueSet;

void Task1Function(void * param)
{
	int i = 0;
	while (1)
	{
		xQueueSend(xQueueHandle1, &i, portMAX_DELAY);//写数据
		i++;
		vTaskDelay(10);
	}
}

void Task2Function(void * param)
{
	int i = -1;
	while (1)
	{
		xQueueSend(xQueueHandle2, &i, portMAX_DELAY);//写数据
		i--;
		vTaskDelay(20);
	}

}

void Task3Function(void * param)
{
	QueueSetMemberHandle_t handle;
	int i;
	while (1)
	{
		/* 读read queue set 哪一个有数据 */
		handle=xQueueSelectFromSet(xQeueSet,portMAX_DELAY);
		/* 读queue */
		xQueueReceive(handle,&i,0);
		printf("get data: %d\r\n",i);
	}
}


/*-----------------------------------------------------------*/

int main( void )
{
	TaskHandle_t xHandleTask1;
		
#ifdef DEBUG
  debug();
#endif

	prvSetupHardware();

	printf("Hello, world!\r\n");

			/*  创建两个队列        */
	xQueueHandle1=xQueueCreate(2,sizeof(int));
	if(xQueueHandle1==NULL)
	{
		printf("can not creat queue\r\n");
	}

	xQueueHandle2=xQueueCreate(2,sizeof(int));
	if(xQueueHandle2==NULL)
	{
		printf("can not creat queue\r\n");
	}
			/*  创建队列集      */
	xQeueSet=xQueueCreateSet(4);//队列A的长度+队列B的长度

			/*  把2个queue添加进queue set  */
	xQueueAddToSet(xQueueHandle1,xQeueSet);
	xQueueAddToSet(xQueueHandle2,xQeueSet);

			/*  创建3个任务 */
	xTaskCreate(Task1Function, "Task1", 100, NULL, 1, &xHandleTask1);
	xTaskCreate(Task2Function, "Task2", 100, NULL, 1, NULL);
	xTaskCreate(Task3Function, "Task3", 100, NULL, 1, NULL);


	/* Start the scheduler. */
	vTaskStartScheduler();

	/* Will only get here if there was not enough heap space to create the
	idle task. */
	return 0;
}

  • 10
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FreeRTOS提供了一种高效的队列机制,可以在任务之间传递数据。下面是使用FreeRTOS队列的步骤: 1. 首先需要创建一个队列。可以使用`xQueueCreate()`函数来创建一个空队列。 ```c QueueHandle_t xQueue; xQueue = xQueueCreate(QUEUE_LENGTH, ITEM_SIZE); ``` 这个函数会返回一个队列句柄,可以用来操作这个队列。其中,QUEUE_LENGTH是队列的长度,ITEM_SIZE是每个队列项的大小(以字节为单位)。 2. 向队列中发送数据。可以使用`xQueueSend()`函数向队列中发送数据。 ```c int data = 123; BaseType_t xStatus; xStatus = xQueueSend(xQueue, &data, portMAX_DELAY); ``` 这个函数的第一个参数是队列句柄,第二个参数是要发送的数据的地址,第三个参数是等待时间,如果队列已满,则等待直到队列有空位。 3. 从队列中接收数据。可以使用`xQueueReceive()`函数从队列中接收数据。 ```c int data; BaseType_t xStatus; xStatus = xQueueReceive(xQueue, &data, portMAX_DELAY); ``` 这个函数的第一个参数是队列句柄,第二个参数是接收数据的地址,第三个参数是等待时间,如果队列为空,则等待直到队列中有数据。 4. 删除队列。当不再需要这个队列时,可以使用`vQueueDelete()`函数删除队列。 ```c vQueueDelete(xQueue); ``` 以上就是FreeRTOS队列的基本使用方法。需要注意的是,在使用队列时要确保发送和接收数据的任务同时存在,否则会出现死锁等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值