事件组
事件组 | 同步点 | |
创建 | xEventGroupCalc = xEventGroupCreate();//1,创建事件组 | xEventGroupSyc = xEventGroupCreate() |
设置 | xEventGroupSetBits(xEventGroupCalc,(1<<0));//设置事件组bit0 位 | xEventGroupSync(xEventGroupSyc,BUSYING,ALL,portMAX_DELAY); //BUSYING = 1 << 0, //ALL = 1 << 0 | 1 << 1 | 1 << 2 |
等待 | xEventGroupWaitBits(xEventGroupCalc,(1<<0)|(1<<1),pdTRUE,pdTRUE,portMAX_DELAY); //xclearonexit:退出是否清除。xwaitforallbits:是否等待所有的事件位有数据 | / |
总结 | 1,事件组解决多个生产者,都在产生数据时,各自往对应的bit位给状态通知。 2,消费者等待不同的bit未是否完成,可以是等待所有位也可以只等待其中一个完成。 3,事件组只能传递完成状态,而不能传递数据,传递数据还要通过队列方式 | 1,同步点,创建同事件组。 2,设置不同于事件组,要传递对应位以及所有需要同步的位的组合。 3,当所有任务都放入事件组,标志位都以完成,才能退出阻塞。 |
1,事件组由一组标志位组成,每一个标志位代表一个特定的时间。
任务可以等待某些标志位被置位或清除。
2,队列,信号量。都是只能共用标志位,无法多标志多个事件。
3,事件组包含,事件创建,生产者1,写bit0,生产者2,写bit1.
消费者等待bit0或bit1。也可以只等待其中一个bit0。
/*
任务1,sum++,sum 每加到1000次后,往队列存放数据,并且记录事件组bit0
任务2,dec--,dec 每减到1000次后,往队列存放数据,并且记录事件组bit1
任务3,等待事件组,bit0,bit1同时标志时,从队列中取出数据打印
*/
static int sum = 0;
static int dec = 0;
static QueueHandle_t qHandle_f1;
static QueueHandle_t qHandle_f2;
static EventGroupHandle_t xEventGroupCalc;
void Task1function(void *param)
{
int i;
while(1)
{
for(i=0;i<1000;i++)
{
sum++;
}
xQueueSend(qHandle_f1, &sum, portMAX_DELAY);
xEventGroupSetBits(xEventGroupCalc,(1<<0));//设置事件组bit0 位
}
}
void Task2function(void *param)
{
int i;
while(1)
{
for(i=0;i<1000;i++)
{
dec--;
}
xQueueSend(qHandle_f2, &dec, portMAX_DELAY);
xEventGroupSetBits(xEventGroupCalc,(1<<1));//设置事件组bit1 位
}
}
void Task3function(void *param)
{
int val_sum;
int val_dec;
while(1)
{
xEventGroupWaitBits(xEventGroupCalc,(1<<0)|(1<<1),pdTRUE,pdTRUE,portMAX_DELAY);
//xclearonexit:退出是否清除。xwaitforallbits:是否等待所有的事件位有数据
xQueueReceive(qHandle_f1,&val_sum,portMAX_DELAY);
xQueueReceive(qHandle_f2,&val_dec,portMAX_DELAY);
printf("sum = %d,dec=%d\r\n",val_sum,val_dec);
}
}
int main( void )
{
prvSetupHardware();
printf("Hello, world!\r\n");
qHandle_f1 = xQueueCreate(10,sizeof(int));
qHandle_f2 = xQueueCreate(10,sizeof(int));
xEventGroupCalc = xEventGroupCreate();//1,创建事件组
xTaskCreate(Task1function, "Task1", 100,NULL, 1, NULL);//
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;
}
同步点
当多个时间同时完成后,执行后面的操作。否则后面的操作处于阻塞状态。
/*
任务1,买菜 buying
任务2,炒菜 cooking
任务3,摆桌 tableing
当3个任务都完成时,吃饭 eating
*/
typedef enum
{
BUYING = 1 << 0,
COOKING = 1 << 1,
TABLEING = 1 << 2,
ALL = 1 << 0 | 1 << 1 | 1 << 2
}eWork_t;
static EventGroupHandle_t xEventGroupSyc;
static SemaphoreHandle_t xSemapMutex;
void Task1function(void *param)
{
while(1)
{
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task1 is buying\r\n");
xSemaphoreGive(xSemapMutex);
xEventGroupSync(xEventGroupSyc,BUYING,ALL,portMAX_DELAY);
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task1 is eating\r\n");
xSemaphoreGive(xSemapMutex);
vTaskDelay(2);
}
}
void Task2function(void *param)
{
while(1)
{
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task2 is cooking\r\n");
xSemaphoreGive(xSemapMutex);
xEventGroupSync(xEventGroupSyc,COOKING,ALL,portMAX_DELAY);
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task2 is eating\r\n");
xSemaphoreGive(xSemapMutex);
vTaskDelay(2);
}
}
void Task3function(void *param)
{
while(1)
{
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task3 is tableing\r\n");
xSemaphoreGive(xSemapMutex);
xEventGroupSync(xEventGroupSyc,TABLEING,ALL,portMAX_DELAY);
xSemaphoreTake(xSemapMutex, portMAX_DELAY);
printf("task3 is eating\r\n");
xSemaphoreGive(xSemapMutex);
vTaskDelay(2);
}
}
int main( void )
{
prvSetupHardware();
printf("Hello, world!\r\n");
xEventGroupSyc = xEventGroupCreate();//创建同步点
xSemapMutex = xSemaphoreCreateMutex();//创建互斥锁
xTaskCreate(Task1function, "Task1", 100,NULL, 3, NULL);//
xTaskCreate(Task2function, "Task2", 100,NULL, 2, 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;
}