FreeRtos入门-4 事件组与同步点

事件组

事件组

同步点

创建

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 theidle 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 theidle task. */return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/792240.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

VB 通过COM接口解析PSD文件

最近有PS测评的需求&#xff0c;故而想到了解析psd文件&#xff0c;目的就是为了获取文档信息和图层信息&#xff1b;获取PS的图像信息有很多方式&#xff0c;有过程性的&#xff0c;比如监听PS的各种操作事件&#xff1b;有结果性的&#xff0c;比如本文写的解析PSD文件。 0.…

使用pip安装geopandas(24.4更新)

geopandas是我们用Python进行地理分析常用的库&#xff0c;在数据处理、分析、制图等场景中有着极为广泛的应用&#xff0c;但是在安装过程中会出现各种问题。​geopandas的安装方式有很多&#xff0c;今天我们选取较为简单的pip来进行geopandas的安装。 ​首先&#xff0c;我…

内部类(InnerClass)

概述 什么是内部类 将一个类A定义在另一个类B里面&#xff0c;里面的那个类A就称为内部类&#xff08;InnerClass&#xff09;&#xff0c;类B则称为外部类&#xff08;OuterClass&#xff09;。 为什么要声明内部类呢 具体来说&#xff0c;当一个事物A的内部&#xff0c;还有…

Java web第一次作业

1.学会用记事本编写jsp文件&#xff0c;并放进tomcat的相关目录下&#xff0c;运行。 源代码&#xff1a; <% page contentType"text/html;charsetUTF-8" language"java" %> <html> <head> <title>我的第一个JSP页面</ti…

JavaSE——运算符

1. 概念 运算符是一种用于执行特定操作的符号或关键字。在编程中&#xff0c;运算符用于对变量、常量和表达式进行操作&#xff0c;以产生一个结果。 作为一门计算机语言&#xff0c; Java 也提供了一套丰富的运算符来操纵变量。 Java 中运算符可分为以下&#xff1a;算术运算…

电商系列之促销

> 插&#xff1a;AI时代&#xff0c;程序员或多或少要了解些人工智能&#xff0c;前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 坚持不懈&#xff0c;越努力越幸运&#xff0c;大家…

算法沉淀——动态规划篇(子数组系列问题(下))

算法沉淀——动态规划篇&#xff08;子数组系列问题&#xff08;下&#xff09;&#xff09; 前言一、等差数列划分二、最长湍流子数组三、单词拆分四、环绕字符串中唯一的子字符串 前言 几乎所有的动态规划问题大致可分为以下5个步骤&#xff0c;后续所有问题分析都将基于此 …

CSS之第一个CSS样式和CSS选择符

前端这些博客&#xff0c;我觉得都是固定的语法&#xff0c;故而不会以过多的文字进行描述&#xff0c;本系列博文均以实例和代码介绍的方式进行&#xff0c;主要按照代码进行。不会以过多的文字描述。 第一个CSS样式 <!DOCTYPE html> <html lang"en">…

【JavaEE初阶系列】——文件操作 IO 之 文件系统操作

目录 &#x1f4dd;认识文件 &#x1f6a9;树型结构组织 和 目录 &#x1f388;绝对路径和相对路径 &#x1f6a9;文件类型 &#x1f4dd;文件系统操作 &#x1f388;File 概述 &#x1f388;File类的使用 1. 绝对路径 vs 相对路径 2. 路径分隔符 3. 静态成员变量 4…

【C语言】翻译环境与运行环境

一、前言 在我们学习C语言的时候&#xff0c;第一个接触的程序就是&#xff1a;在屏幕上打印” hello word! “&#xff0c;可当时的我们却未去深入的理解与感悟&#xff0c;一个程序代码是如何运行的&#xff1b;而这一期的博客&#xff0c;则是带着我们&#xff0c;通过C代码…

mac电脑安装redis教程

1、下载地址 Download | RedisRedisYou can download the last Redis source files here. For additional options, see the Redis downloads section below.Stable (7.2)Redis 7.2 …https://redis.io/download/#redis-downloads 2、安装 2.1 解压下载后的压缩文件 2.2 进入…

Vulnhub:WESTWILD: 1.1

目录 信息收集 arp nmap nikto whatweb WEB web信息收集 dirmap enm4ulinux sumbclient get flag1 ssh登录 提权 横向移动 get root 信息收集 arp ┌──(root㉿ru)-[~/kali/vulnhub] └─# arp-scan -l Interface: eth0, type: EN10MB, MAC: 0…

LeetCode-236. 二叉树的最近公共祖先【树 深度优先搜索 二叉树】

LeetCode-236. 二叉树的最近公共祖先【树 深度优先搜索 二叉树】 题目描述&#xff1a;解题思路一&#xff1a;递归判断解题思路二&#xff1a;0解题思路三&#xff1a;0 题目描述&#xff1a; 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖…

linux操作系统的进程状态

这个博客只是为了自己复习用的&#xff01;&#xff01;&#xff01; 冯诺依曼体系结构 计算机是由一个一个硬件组成的 输入设备&#xff1a;键盘&#xff0c;鼠标&#xff0c;扫描仪&#xff0c;写板等等 中央处理器&#xff08;CPU&#xff09;:含有运算器和控制器等 输出单…

【算法练习】27:冒泡排序学习笔记

一、冒泡排序的算法思想 原理&#xff1a;以升序为例&#xff0c;冒泡排序通过从左往右连续比较相邻元素&#xff0c;当发现左边比右边大就交换元素。从左往右依次比较完称为“一轮”&#xff0c;每轮结束之后就会固定一个元素。 时间复杂度&#xff1a;2层循环&#xff0c;所以…

不讲概念,讲实操,mysql 分表模糊查询、分页查询 及 merge 表的使用

1.Mysql merge合并表的要求 1.合并的分表必须是 MyISAM 引擎&#xff0c;MyISAN引擎是不支持事务的。2.Merge表只保证合表后数据唯一性&#xff0c;合表前的数据可能会存在重复。3.表的结构必须一致&#xff0c;包括索引、字段类型、引擎和字符集。4.删除 tb_member1 分表正确…

Python实现BOA蝴蝶优化算法优化卷积神经网络分类模型(CNN分类算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 蝴蝶优化算法(butterfly optimization algorithm, BOA)是Arora 等人于2019年提出的一种元启发式智能算…

[技术闲聊]我对电路设计的理解(三)

终于可以独立做项目了&#xff0c;是不是很激动&#xff0c;是不是为自己骄傲和自豪&#xff0c;应该的&#xff0c;奋斗那么久不就是为了站在山巅看看四周的风景嘛&#xff01; 虽说山外还有山&#xff0c;但是此刻就在脚下的山巅上&#xff0c;怡然自得都是不过分的&#xff…

LLM端侧部署系列 | 如何将阿里千问大模型Qwen部署到手机上?实战演示(下篇)

引言 简介 编译Android可用的模型 转换权重 生成配置文件 模型编译 编译apk 修改配置文件 绑定android library 配置gradle 编译apk 手机上运行 安装 APK 植入模型 效果实测 0. 引言 清明时节雨纷纷&#xff0c;路上行人欲断魂。 小伙伴们好&#xff0c;我是《小…

9.动态规划——4.最长公共子序列(动态规划类的算法题该如何解决?)

例题——最长公共子序列(一) 分析 设最长公共子序列 d p [ i ] [ j ] dp[i][j] dp[i][j]是 S 1 S_1 S1​的前 i i i个元素&#xff0c;是 S 2 S_2 S2​的前 j j j个元素&#xff0c;那么有&#xff1a; 若 S 1 [ i − 1 ] S 2 [ i − 1 ] S_1[i-1]S_2[i-1] S1​[i−1]S2​[…