1事件标志组概念
事件标志组:是一组事件标志位的集合, 可以简单的理解事件标志组,就是一个整数。
 其特点:
 1)它的每一个位表示一个事件(高8位不算)
 2)每一位事件的含义,由用户自己决定,如:bit0表示按键是否按下,bit1表示是否接受到消息,值为1:表示事件发生了;值为0:表示事件未发生
 3)任意任务或中断都可以读写这些位
 4)可以等待某一位成立,或者等待多位同时成立
2 事件标志组变量定义
typedef TickType_t EventBits_t;
#if ( configUSE_16_BIT_TICKS  = =  1 )typedef   uint16_t   TickType_t;
#elsetypedef   uint32_t   TickType_t;
#endif
#define  configUSE_16_BIT_TICKS    0 
EventBits_t 实际上是一个 16 位或 32 位无符号的数据类型
 注意:一个事件组最多可以存储 24 个事件标志!
3 事件标志组与队列、信号量的比较

4 事件标志组API函数

 1)创建事件标志位
 EventGroupHandle_t xEventGroupCreate ( void )
 返回值:NULL 失败
 其他 成功返回句柄
 2)清零时间标志位
 EventBits_t xEventGroupClearBits( EventGroupHandle_t xEventGroup, 
 const EventBits_t uxBitsToClear ) 
 形参: xEventGroup 事件标志组
 uxBitsToClear 待清零的事件标志位
 返回值:整数 清零事件标志位之前事件组中事件标志位的值
 3)设置事件标志位
 EventBits_t xEventGroupSetBits( EventGroupHandle_t xEventGroup, 
 const EventBits_t uxBitsToSet )
 形参: xEventGroup 事件标志组
 uxBitsToSet 待设置的事件标志位
 返回值:整数 函数返回时,事件标志位值
 4)等待事件标志位
 EventBits_t xEventGroupWaitBits( EventGroupHandle_t xEventGroup,
 const EventBits_t uxBitsToWaitFor,
 const BaseType_t xClearOnExit,
 const BaseType_t xWaitForAllBits,
 TickType_t xTicksToWait )
 形参: xEventGroup 事件标志组句柄
 uxBitsToWaitFor 等待的事件标志位,可以用逻辑或等待多个事件标志位
 xClearOnExit 成功等待到事件标志位后,清除事件组中对应的事件标志位,
 pdTRUE :清除uxBitsToWaitFor指定位;
 pdFALSE:不清除
 xWaitForAllBits 等待 uxBitsToWaitFor 中的所有事件标志位(逻辑与)
 pdTRUE:等待的位,全部为1
 pdFALSE:等待的位,某个为1
 xTicksToWait 等待的阻塞时间
返回值:等待的事件标志位值 等待事件标志位成功,返回等待到的事件标志位
 其他值 等待事件标志位失败,返回事件组中的事件标志位
注意:
 可以等一位,也可以等多位
 等到之后,可以清楚某些位
 5)同步函数
 EventBits_t xEventGroupSync( EventGroupHandle_t xEventGroup, 
 const EventBits_t uxBitsToSet, 
 const EventBits_t uxBitsToWaitFor, 
 TickType_t xTicksToWait)
  
 
5事件标志组实战
 
5.1freertos_demo.c
#include "freertos_demo.h"
#include "./BSP/LCD/lcd.h"
#include "./BSP/KEY/key.h"
/*FreeRTOS*********************************************************************************************/
#include "FreeRTOS.h"
#include "task.h"
#include "event_groups.h"/******************************************************************************************************/
/*FreeRTOS配置*//* START_TASK 任务 配置* 包括: 任务句柄 任务优先级 堆栈大小 创建任务*/
#define START_TASK_PRIO 1                   /* 任务优先级 */
#define START_STK_SIZE  128                 /* 任务堆栈大小 */
TaskHandle_t            StartTask_Handler;  /* 任务句柄 */
void start_task(void *pvParameters);        /* 任务函数 *///1.FreeRTOS配置
//1.1 START_TASK 任务 配置
//包括: 任务句柄 任务优先级 堆栈大小 创建任务#define START_TASK_PRIO 1                   /* 任务优先级 */
#define START_STK_SIZE  128                 /* 任务堆栈大小 */
TaskHandle_t            StartTask_Handler;  /* 任务句柄 */
void start_task(void *pvParameters);        /* 任务函数 *///1.2 TASK1 任务 配置
//包括: 任务句柄 任务优先级 堆栈大小 创建任务#define TASK1_PRIO      2                   /* 任务优先级 */
#define TASK1_STK_SIZE  128                 /* 任务堆栈大小 */
TaskHandle_t            Task1Task_Handler;  /* 任务句柄 */
void task1(void *pvParameters);             /* 任务函数 *///1.3 TASK2 任务 配置
//包括: 任务句柄 任务优先级 堆栈大小 创建任务#define TASK2_PRIO      4                   /* 任务优先级 */
#define TASK2_STK_SIZE  128                 /* 任务堆栈大小 */
TaskHandle_t            Task2Task_Handler;  /* 任务句柄 */
void task2(void *pvParameters);             /* 任务函数 *///1.4 TASK3 任务 配置
//包括: 任务句柄 任务优先级 堆栈大小 创建任务#define TASK3_PRIO      3                   /* 任务优先级 */
#define TASK3_STK_SIZE  128                 /* 任务堆栈大小 */
TaskHandle_t            Task3Task_Handler;  /* 任务句柄 */
void task3(void *pvParameters);             /* 任务函数 *///1.5 定义事件组句柄、将事件组位0、位1设置为1
EventGroupHandle_t   eventgroup_handle;
#define EVENTBIT_0    (1 << 0 )
#define EVENTBIT_1    (1 << 1 )//2 FreeRTOS例程入口函数
// @param       无
//@retval      无
void freertos_demo(void)
{//创建事件组eventgroup_handle = xEventGroupCreate ();if(eventgroup_handle != NULL){printf("事件标志组创建成功!!\r\n");}        xTaskCreate((TaskFunction_t )start_task,            /* 任务函数 */(const char*    )"start_task",          /* 任务名称 */(uint16_t       )START_STK_SIZE,        /* 任务堆栈大小 */(void*          )NULL,                  /* 传入给任务函数的参数 */(UBaseType_t    )START_TASK_PRIO,       /* 任务优先级 */(TaskHandle_t*  )&StartTask_Handler);   /* 任务句柄 */vTaskStartScheduler();
}//2 start_task
// @param       pvParameters : 传入参数(未用到)
//@retval      无
void start_task(void *pvParameters)
{taskENTER_CRITICAL();           /* 进入临界区 *//* 创建任务1 */xTaskCreate((TaskFunction_t )task1,(const char*    )"task1",(uint16_t       )TASK1_STK_SIZE,(void*          )NULL,(UBaseType_t    )TASK1_PRIO,(TaskHandle_t*  )&Task1Task_Handler);/* 创建任务2 */xTaskCreate((TaskFunction_t )task2,(const char*    )"task2",(uint16_t       )TASK2_STK_SIZE,(void*          )NULL,(UBaseType_t    )TASK2_PRIO,(TaskHandle_t*  )&Task2Task_Handler);vTaskDelete(StartTask_Handler); /* 删除开始任务 */taskEXIT_CRITICAL();            /* 退出临界区 */
}//3 task1,释放二值信号量
//@param       pvParameters : 传入参数(未用到)
// @retval      无
void task1(void *pvParameters)
{uint8_t key = 0;while (1){key = key_scan(0);       //扫描事件标志组if(key == KEY0_PRES){xEventGroupSetBits(eventgroup_handle,EVENTBIT_0);      //将事件标志组的bit 0位置置为1}else if(key == KEY1_PRES){xEventGroupSetBits(eventgroup_handle,EVENTBIT_1);      //将事件标志组的bit 1位置置为1}vTaskDelay(10);}
}//task2
//@param       pvParameters : 传入参数(未用到)
//@retval      无
void task2(void *pvParameters)
{EventBits_t event_bit = 0;while (1){event_bit = xEventGroupWaitBits(eventgroup_handle,    //事件标志组句柄EVENTBIT_0 | EVENTBIT_1,          //等待事件标志组bit0和bit1位pdTRUE,                           //成功等到事件标志位后,清除事件标志组中bit0 bit1位pdTRUE,                           //等到事件标志组bit0 bit1位 都置1成立portMAX_DELAY);                    //死等printf("等待的事件标志位:%#x\r\n",event_bit);}
}