rtos任务间通信为什么不用全局变量?

strongerHuang strongerHuang     2022-12-07     424

关键词:

关注+星标公众,不错过精彩内容

作者 | strongerHuang

微信公众号 | strongerHuang

有深入理解RTOS原理,或阅读过RTOS源码的同学应该知道:RTOS实现任务间通信通常是由一系列指针进行操作实现的。

任务间通信的“有效数据”,其实也是由指针指向一个“变量”或“数组”实现的。

1.信号量

信号量,本质是传递一个“事件”。比如:任务A完成发送数据,通过信号量通知任务B。

OSSemPost(EventSem_SendOK);

我们主要想传递“完成发送数据”这个“事件”,进一步分析,其实就是一个“标志”或“变量”

2.队列

队列和信号量原理类似有点类似,只是这里是“变量”。比如:串口接收完成一帧数据,通过队列发送给任务B.

OSQPost(UARTRcvQueue, RcvBuf);

相比信号量,队列传递的数据量更大,队列传递的有效数据一般是“数组”。

还有邮箱,与队列类似,可以理解为“二维数组”。

写到这里,你会发现,不管信号量,还是队列,底层本质也是传递“变量”“数组”。

那么问题来了:RTOS任务间通信为什么不用全局变量?

这个问题比较常见,也看到在我的技术交流群有讨论,所以就简单来分享一下看法。

全局变量有什么问题?

RTOS任务间通信为什么不用全局变量?原因在于使用全局变量存在诸多弊端。

1.抢占问题

两个或多个任务,都要去“使用”同一个全局变量,如果不添加任何“互斥”措施,必定会存在抢占的问题。

2.代码规范问题

整个项目只有少数几个全局变量没什么问题,如果是整个项目有几十个,甚至几百个全局变量,你觉得这样的代码,后面好维护吗?

经过多次迭代,代码只会越来越难理解,越来越难阅读。

3.耦合性问题

全局变量会导致分层不合理与模块化编程相违背,你的全局变量没有归属,既不是任务A,也不是模块A,最终可能“任人宰割”导致“夭折”。

4.安全性问题

有一句话怎么说的呢,全局变量是项目的“罪魁祸首”,项目做大之后,一旦有小修改,可能就会引发大Bug.

全局变量还有很多弊端,这里就不一一描述了,总之一点:慎用全局变量

当然,以上描述的问题(弊端)都是基于项目中存在多个变量的情况,如果项目只有1、2个全局变量,这种不在本文讨论范围之内。

信号量、队列通信原理

大部分RTOS的信号量、队列都是使用指针、结构体、数组等,结合系统进行“封装”,使任务间通信更加有效、安全,同时也遵循“高内聚低耦合”的原则。

比如ucos的信号量post:

INT8U  OSSemPost (OS_EVENT *pevent)

#if OS_CRITICAL_METHOD == 3u                          /* Allocate storage for CPU status register      */
    OS_CPU_SR  cpu_sr = 0u;
#endif


#if OS_ARG_CHK_EN > 0u
    if (pevent == (OS_EVENT *)0)                     /* Validate 'pevent'                             */
        return (OS_ERR_PEVENT_NULL);
    
#endif
    if (pevent->OSEventType != OS_EVENT_TYPE_SEM)    /* Validate event block type                     */
        return (OS_ERR_EVENT_TYPE);
    
    OS_ENTER_CRITICAL();
    if (pevent->OSEventGrp != 0u)                    /* See if any task waiting for semaphore         */
                                                      /* Ready HPT waiting on event                    */
        (void)OS_EventTaskRdy(pevent, (void *)0, OS_STAT_SEM, OS_STAT_PEND_OK);
        OS_EXIT_CRITICAL();
        OS_Sched();                                   /* Find HPT ready to run                         */
        return (OS_ERR_NONE);
    
    if (pevent->OSEventCnt < 65535u)                 /* Make sure semaphore will not overflow         */
        pevent->OSEventCnt++;                         /* Increment semaphore count to register event   */
        OS_EXIT_CRITICAL();
        return (OS_ERR_NONE);
    
    OS_EXIT_CRITICAL();                               /* Semaphore value has reached its maximum       */
    return (OS_ERR_SEM_OVF);

我们需要传递的有效信息虽然只有一个变量,但它会做“临界区”管理,以及预判一些错误的情况等。

最后,RTOS源码也可以算是一个优秀的项目,特别是目前普及率比较高、装机量比较多的RTOS,比如µC/OS、FreeRTOS、RT-Thread、ThreadX等。

最最后,有时间的小伙伴可以阅读一下RTOS源码,RTOS内核我推荐µC/OS,阅读源码能让你掌握一些软件架构的知识,也能让你明白一些开发过程种常见的问题。

------------ END ------------


●专栏《嵌入式工具

●专栏《嵌入式开发》

●专栏《Keil教程》

●嵌入式专栏精选教程

关注公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。


点击“阅读原文”查看更多分享。

rtos内功修炼记——任务入口函数执行完毕之后去哪里?(代码片段)

...-任务间同步机制的实现,万变不离其宗06-任务间通信为什么不用全局变量07-动态内存和静态内存管理机制08-C 查看详情

ucosiii消息队列(代码片段)

1、UCOSIII任务间通信1.1、任务间通信:一个任务或者中断服务程序有时候需要和另一个任务交流信息,这个就是消息传递的过程就叫做任务间通信,任务间的消息传递可以通过2种途径:一是全局变量,二是通... 查看详情

freertos事件标志组

为什么要使用事件标志事件标志组是实现多任务同步的有效机制之一。也许有不理解的初学者会问采用事件标志组多麻烦,搞个全局变量不是更简单?其实不然,在裸机编程时,使用全局变量的确比较方便,但是在加上RTOS后就... 查看详情

freertos-事件标志组,消息队列,信号量,二值信号量,互斥信号量

任务间的通信和同步机制 在裸机编程时,使用全局变量的确比较方便,但是在加上RTOS后就是另一种情况了。使用全局变量相比事件标志组主要有如下三个问题:1、使用事件标志组可以让RTOS内核有效地管理任务,而全局变量... 查看详情

ucosiii消息队列(代码片段)

1、UCOSIII任务间通信1.1、任务间通信:一个任务或者中断服务程序有时候需要和另一个任务交流信息,这个就是消息传递的过程就叫做任务间通信,任务间的消息传递可以通过2种途径:一是全局变量,二是通... 查看详情

基于rtos的软件开发理论(代码片段)

基于RTOS的软件开发理论RTOS的特点任务设计任务的特性任务划分的方法设备依赖性任务关键任务紧迫任务数据处理任务触发条件相同的任务运行周期相同的任务顺序操作任务任务的可调度性分析任务类型单次执行的任务周期性执... 查看详情

是否可以为进程间通信提供共享全局变量?

】是否可以为进程间通信提供共享全局变量?【英文标题】:Isitpossibletohaveasharedglobalvariableforinter-processcommunication?【发布时间】:2020-05-2807:10:16【问题描述】:我需要为我的操作系统类解决并发分配问题。我不想要这里的解决方... 查看详情

vc++进程间的通信

...言,应用程序中的一个次要线程总是为主线程执行特定的任务,这样,主线程和次要线程间必定有一个信息传递的渠道,也就是主线程和次要线程间要进行通信。这种线程间的通信不但是难以避免的,而且在多线程编程中也是复杂和... 查看详情

线程间的通信同步方式与进程间通信方式

1.线程间的通信方式1)使用全局变量(由于多个线程可能更改全局变量,因此全局变量最好声明为volatile)2)使用消息实现通信 3)使用事件CEvent类实现线程间的通信2.同步/异步(C端)同步:在C端发出一个功能调用时,没有得到结果之前... 查看详情

嵌入式开发基础之线程间通信

线程间通信引言基本概念邮箱消息队列消息队列的基本概念消息队列的工作机制后续引言在裸机编程中,经常会使用全局变量进行功能间的通信,如某些功能可能由于一些操作而改变全局变量的值,另一个功能对此全... 查看详情

rtos任务间互斥的问题

...目时,通常都会遇到互斥的情况,比如:几个任务都要使用一个UART串口进行发送数据。如果不加互斥锁,优先级高的任务,会抢占串口并发送数据,则有可能会出现发送数据“乱码”的情况。今天就说 查看详情

rtos任务间互斥的问题

...目时,通常都会遇到互斥的情况,比如:几个任务都要使用一个UART串口进行发送数据。如果不加互斥锁,优先级高的任务,会抢占串口并发送数据,则有可能会出现发送数据“乱码”的情况。今天就说 查看详情

线程间通信常用的三种方法

多线程通信的方法主要有以下三种: 1.全局变量进程中的线程间内存共享,这是比较常用的通信方式和交互方式。注:定义全局变量时最好使用volatile来定义,以防编译器对此变量进行优化。2.Message消息机制常用的Message通信... 查看详情

基于rtos的软件开发理论(代码片段)

基于RTOS的软件开发理论RTOS的特点任务设计任务的特性任务划分的方法设备依赖性任务关键任务紧迫任务数据处理任务触发条件相同的任务运行周期相同的任务顺序操作任务任务的可调度性分析任务类型单次执行的任务周期性执... 查看详情

多任务实时系统中的同步与通信

在多任务实时系统中,常常需要在任务之间或者中断与任务之间进行通信,这就产生了同步与通信机制。同步可分为两种:①资源同步:避免两个及以上任务对同一个资源的同时操作②活动同步:确定任务的活动是否到达一个确... 查看详情

大疆面试整理

...言,应用程序中的一个次要线程总是为主线程执行特定的任务,这样,主线程和次要线程间必定有一个信息传递的渠道,也就是主线程和次要线程间要进行通信。这种线程间的通信不但是难以避免的,而且在多线程编程中也是复杂和... 查看详情

freertosfreertos学习笔记(14)—freertos的消息队列(原生api)(代码片段)

...是就是用队列来传递消息。队列就是一个数据结构,用于任务间的数据的传递。一、消息队列的基本概念队列又称消息队列,是一种常用于任务间通信的数据结构,队列可以在任务与任务间、中断和任务间传递信息,实现了任务... 查看详情

等待与唤醒案例

...个是消费,那么线程A与线程B之间就存在线程通信问题。为什么要处理线程间通信:多个线程并发执行时,在默认情况下CPU是随机切换线程的,当我们需要多个线程来共同完成一件任务,并且我们希望他们有规律的执行,那么多线... 查看详情