为全局变量赋值_实例分析如何远离漫天飞舞的全局变量

前篇《由static来谈谈模块封装》基本实现了对外隐藏属性,隐藏局部模块函数,开放接口的功能。对于这个话题还有些点没有深入探讨:为什么要这样做?以及这样做的好处。或许很多刚刚开始用C或者其他面向对象编程语言(比如C++)的小伙伴们,常常在一个项目里为了图省事,整了很多全局对象、全局变量满天飞,这样做其实是有很多弊端,本文来聊聊这个话题。

先谈谈全局变量的特点

全局变量(Global Variables):在计算机编程语言中,所谓全局变量是指具有全局作用域的变量,这意味着它在整个程序中是可见的,因此是可访问的。所谓可访问,是指全局可读、全局可写。在编译语言中,全局变量通常是静态变量,其范围(生命周期)是程序的整个运行时。当然解释性语言除外,解释性语言包括命令行解释器(比如python, Java script,shell等)中,全局变量通常在声明时由解释器动态分配,这是由于解释性语言是读取>解释>执行模式,不像编译性语言,运行前可预知变量属性,解释性语言读取解释前无从获取变量属性。

在C/C++编程语言中,全局变量的这种全局可见性特点,滥用全局变量会让代码表现当相当邪恶!如果使用全局变量,就意味着下面这些场景的存在:

  • 实际代码可能有很多地方在读、在写全局变量
  • 全局变量在多线程或多任务间共享
  • 全局变量在常规代码和中断服务程序间共享

为啥说全局变量很邪恶?

单片机裸机编程

或许你会说,我就这样用?咋了?软件也跑的很好啊?来看看这个场景:

一个超字宽的变量(比如16位单片机,字宽即为16位),正被一个常规代码在写变量数据域时且还没写完,啪叽,来了个中断!中断一来,CPU赶紧把手里的活儿停下来,奔过去处理中断了,不巧在中断函数里,该变量因业务需求有需要写这个变量有经验的不这么写,仅为了方便说明:

举个栗子,还是以之前文章的传感器为例,实际应用中传感器可能是下面这样的数据结构来描述:

#ifndef _SENSOR_H_#define _SENSOR_H_typedef struct _t_sensor{/* 测量值与测量范围及单位有关 */float value;/* 测量范围,根据采样值映射  */float upper_range;float lower_range;/* 温度单位 */unsiged char unit;
}T_SENSOR;/*假定是一个温度测量产品*/extern T_SENSOR temperature;#endif _SENSOR_H_

假定这个传感器数据结构有这样一些被访问的可能:

  1. 上位机会改写测量数据的范围及单位,串口通信中断服务程序直接写这个全局变量中的上下限数据域
  2. LCD操作界面可改写温度上下限范围。
  3. 测量更新模块根据当前范围及单位配置,将传感器采集到的数据映射为测量值。

这些需求用例,用图描述一下:

2dd997188f4f979434e4c83ab96530c0.png

比如用户操作HMI界面正改写温度范围,而此时远程上位机也正改写温度范围,按上面这个做法,可能出现哪些邪恶的后果呢?

  1. 通过LCD界面写入上限为300.5(假定原下限为0),此时远程串口报文收到,程序直接在中断服务程序将范围修改为(-100,200.5),此时中断返回,用户可能接着修改下限为-200,则最终设备内的温度范围可能既不是(-100,200.5)也不是(-200,300.5),而可能是(-200,200.5)。这是一个易理解的数据混乱的场景。
  2. 现实中如果使用的单片机是8位/16位单片机,一条指令无法完成操作一个32位立即数,有可能才完成一个浮点数中某几个字节,此时就被中断打断写入200,然后中断返回后继续写入剩下字节,数据可能会变得非常诡异!利用http://www.speedfly.cn/tools/hexconvert/ 在线工具转换浮点数到16进制:
0x43964000 /* 浮点数300.5的16进制*/
0x43488000 /* 浮点数200.5的16进制*/

假定中断进入时,HMI界面程序写入了0x4396前两个字节,中断返回时,上限改写为200.5(0x43488000),此时继续执行后面两个字节写入,则上限变成为(0x43484000),来看看这个数是多大?变成了200.25,这是不是很邪恶?

c9e3f0df9a2eadc987fbb43918794b01.png

或许有的朋友会说,可以在LCD写范围时关中断嘛。诚然,可以这么做:

void hmi_operate(){
    /*关中断*/
    _disable_interrupt();
    /*改写温度范围*/
    ....
    /*开中断*/
    _enable_interrupt();
}

但是如果这个全局变量有很多地方在改写,为了数据安全,势必就到处开/关中断,这样做的坏处:

  • 经常开关中断,势必影响中断响应,会有概率丢失异步中断处理(比如串口按字节接收中断,可能就会漏收字节),程序不健壮,工作不稳定。
  • 到处访问改写,不易调试,群魔乱舞,代码也不易维护。想加点东西,改点东西可能随处都是坑,一不小心就掉坑里去了!
  • 初学者甚至不会用struct将相关的数据包在一起,其结果是代码里到处都是基本类型的全局变量。一些简单的业务逻辑实现变成一个复杂的代码,数据信息流向一团乱麻。
a8f497d6d75fa72f05834fec5d51b085.gif

裸机程序策略

对于上面这样一个应用场景,怎么解决这种混乱的现象呢。这里分享一下我的思路,这里将主要的串口以及测量模块的设计思路用UML图描述一下大体思路:

722d21eb0c4500ddcce90cdc553e6300.png

如此一来,外部就看不到全局变量了,只需要调用对应的set/get方法即可实现读写访问,由于是裸机前后台程序,数据流向就变的非常清晰了。main函数的主循环大致就可能是这样:

void main(void){
   /*模块初始化*/
   init_uart();
   init_temperature();
   ....
   
   while(1)
   {
       interprete_uart();
       /*可能是周期性调用*/
       if(timer_100ms)
       {
          timer_100ms = 0;
          update_temperature();
       }        
         
       ....
   }   
}

那么uart协议解析要怎么做呢?

void interprete_uart(void){
    if(rx_msg.flag)
    {
        rx_msg.flag = false;
        /*报文完整性检查*/
        ...
            
        /*设置温度配置*/
        set_upper_range(xxx);
        set_lower_range(xxx);
        set_unit(xxx);
    }
    
    if(tx_msg.flag)
    {
        tx_msg.flag = false;
        start_send();
    }
}

static start_send(T_UART_MSG *pMsg){
    /*负责底层操作,启动中断传输*/
}

/*提供应答数据接口*/
void reply_temperature_setting(T_SENSOR sensor){
    /*解析传入参数并封装应答报文*/
}

如此一来,数据流向将变得很清晰,串口接收到数据更新范围配置时,也无需开关中断了,从应用角度几乎见不到全局变量。当然这样做的代价就是会增加一些栈开销。但是这种代价还是值得的。

对于测量模块的set函数思路稍做说明:

int set_upper_range(float range){
    T_SENSOR temp = temperature;
    temp.upper_range = range;
    /*实现范围合理性检查*/
    if(check_range(temp))
    {
        /*两个结构体变量可以直接赋值*/
        temperature = temp;
        return 0;
    }
    else
    {
       return -1;
    }
}

int set_unit(E_UNIT unit){
    if(unit>E_UNIT_F)
        return -1;
    adjust_range(&temperature,unit);
    temperature.unit = unit;    
}

上述代码旨在分享个人的一些思路,其中或有不够严谨的地方,但通过这样的设计思路,应能大幅度远离满天飞的全局变量。

多任务/多线程环境

上面描述其实本质上描述了裸机程序里,普通模式运行程序与中断服务程序对于临界资源的竞争。事实上现在不管是单片机,还是处理器,大多都是基于一个操作系统进行应用开发。甚至还可能是多核芯片,这里就存在并发竞争访问资源的问题。

临界资源:各任务/线程采取互斥的方式,实现共享的资源称作临界资源。属于临界资源的硬件串口打印、显示等,软件有消息缓冲队列、变量、数组、缓冲区等。多任务/线程间应采取互斥方式,从而实现对这种资源的共享。

多任务/多线程情况下在写模块时,只需要封装进保护机制即可。常见的保护机制有关中断、信号量、互斥锁等。在Linux内核中为应对多核并发访问还有自旋锁机制。由于篇幅所限,本文就不做展开了,先挖个坑,以后有机会再分享吧。

总结一下

在前文介绍static文章的基础上,相对更深入的介绍了为何需要隐藏属性以及开放接口的做法。以及如何远离邪恶的全局变量漫天飞舞的不良设计风格。

辛苦原创总结,如果觉得有价值也请帮忙点赞/在看/转发支持,不胜感激!

END

往期精彩推荐,点击即可阅读

猜你喜欢

干货 | 嵌入式必备技能之Git的使用

CPU中的程序是怎么运行起来的

嵌入式系统软件架构设计

Linux下应用开发基础

【Linux笔记】Pinctrl子系统与GPIO子系统

ad7887cedc83570591fce4aa649193e0.png

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

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

相关文章

数据库班级字段怎么定义名称_班级名称

数据库班级字段怎么定义名称在Java中,每个类都有一个名称。 类位于包中,这使我们程序员可以一起工作,避免名称冲突。 我可以命名我的班级A ,也可以命名您的班级A ,只要它们位于不同的程序包中,它们可以很好…

计算机指令取决,不同的计算机,其指令不同,这主要取决于什么?

不同的计算机,其指令系统也不同,这主要取决于所用的CPU。1、CPU指中央处理器,是一块超大规模的集成电路,是一台计算机的运算核心(Core)和控制核心( Control Unit)。它的功能主要是解释计算机指令以及处理计算机软件中的数据。2、程…

python内存管理方法_Python 内存管理大揭秘

前言语言的内存管理是语言设计的一个重要方面。它是决定语言性能的重要因素。无论是C语言的手工管理,还是Java的垃圾回收,都成为语言最重要的特征。这里以Python语言为例子,说明一门动态类型的、面向对象的语言的内存管理方式。对象的内存使用…

Kubernetes集群上的Apache Ignite和Spring第1部分:Spring Boot应用程序

在之前的一系列博客中,我们在Kubernetes集群上启动了一个Ignite集群。 在本教程中,我们将使用先前在Spring Boot Application上创建的Ignite集群。 让我们使用Spring Boot创建我们的项目。 Spring Boot应用程序将连接到Ignite集群。 让我们添加依赖项…

计算机VB整除,【原创】VB中的整除运算和转换函数

前言&#xff1a;【关于四舍五入】——实际上是相对小数部分<0.5 舍&#xff1b;0.5 看整数部分&#xff0c;若为奇数则进&#xff0c;若为偶数则舍&#xff1b;>0.5 进整除运算“\”作用&#xff1a;用于对两个数进行除法运算并返回一个整数例如&#xff1a;18\53 …

栈空间_Linux中的进程栈和线程栈

1. 进程栈进程栈是属于用户态栈&#xff0c;和进程虚拟地址空间 (Virtual Address Space) 密切相关。那我们先了解下什么是虚拟地址空间&#xff1a;在 32 位机器下&#xff0c;虚拟地址空间大小为 4G。这些虚拟地址通过页表 (Page Table) 映射到物理内存&#xff0c;页表由操作…

mockito_Mockito 101

mockitoMockito是一个模拟框架&#xff0c;可让您使用简洁的API编写漂亮的测试。 它偏向于最小的规格&#xff0c;使不同的行为看起来有所不同&#xff0c;并显示清晰的错误消息。 创造嘲弄 要使用Mockito创建模拟&#xff0c;只需使用Mock注释模拟&#xff0c;然后调用Mockit…

csgo显示服务器失败,csgo服务器失败

csgo服务器失败 内容精选换一换您可以通过“应用管理”页面的应用列表&#xff0c;快速查看应用状态&#xff0c;及相关异常信息&#xff0c;如图1所示。包括&#xff1a;应用状态&#xff1a;即图1中的①应用异常信息&#xff1a;即图1中的②云服务器异常信息&#xff1a;即图…

mysql多表成绩查询_MySQL多表数据记录查询(一)

1&#xff0e;交叉连接SQL语句的语法结构如下&#xff1a;select * from表1 cross join 表2;或Select * from表1&#xff0c;表2;2.内连接SQL语句有两种表示形式&#xff1a;使用inner join 语法结构如下&#xff1a;Select表达式1&#xff0c;表达式2&#xff0c;...&#xff…

DMN中的函数式编程:感觉就像再次重读我的大学课程一样

在本文中&#xff0c;我想分享有关DMN中的递归支持的有趣见解&#xff0c;并重点介绍FEEL语言的特定属性如何使功能编程结构能够在DMN中建模。 我们将从一个基本的示例开始&#xff0c;以演示FEEL语言和DMN构造的“商业友好”性质如何使我们能够解决一个通常不愉快的问题&…

手游极品飞车无限狂飙链接服务器失败,极品飞车无极限无法联网是什么原因 联网失败原因分析及解决方法...

有些玩家对于极品飞车无极限游戏中无法联网的问题而困扰&#xff0c;应该怎么解决呢&#xff1f;下面42824小小编就把方法分享给大家&#xff01;一、极品飞车无极限游戏无法联网原因及解决方法1、网络连接不稳定推荐在wifi的情况下进行游戏&#xff0c;如果是3G网的话很容易会…

mysql索引命名规范_mysql使用规范-索引规范

(1)单张表中索引数量不超过5个。(2)单个索引中的字段数不超过5个。(3)索引名必须全部使用小写。(4)非唯一索引按照“idx_字段名称[_字段名称]”进用行命名。例如idx_age_name。(5)唯一索引按照“uniq_字段名称[_字段名称]”进用行命名。例如uniq_age_name。(6)组合索引建议包含…

junit规则_jUnit:规则

junit规则规则在测试&#xff0c;测试用例或测试套件周围增加了特殊处理。 他们可以对该类中的所有测试执行通用的其他验证&#xff0c;并发运行多个测试实例&#xff0c;在每个测试或测试用例之前设置资源&#xff0c;然后在之后拆除它们。 该规则可以完全控制将要应用到的测…

mysql中创建唯一索引的关键字_mysql中唯一索引的关键字是什么

mysql中唯一索引的关键字是unique index。创建唯一索引可以避免数据出现重复。唯一索引可以有多个&#xff0c;但索引列的值必须唯一&#xff0c;索引列的值允许有空值。创建唯一索引可以使用关键字UNIQUE随表一同创建。mysql中唯一索引的关键字是unique index。(推荐教程&…

奇迹觉醒qq服务器比微信少,十年内最大的奇迹!功能比QQ还少的微信为什么能成功?...

今天&#xff0c;微信迎来了自己2021年的第一次「翻车」——2021年1月18日下午2点前后&#xff0c;「由于系统抖动原因」部分微信用户无法及时收取微信消息。截止下午3点19分&#xff0c;故障已被修复。其实微信曾面临过几次信息服务中断的事故&#xff1a;2013年&#xff0c;微…

从fork-join /线程池调用的Singelton bean中的访问spring请求范围缓存

问题&#xff1a; 启用了Spring且将范围设置为Request的缓存需要由不在请求范围内的singleton bean访问。 解&#xff1a; Spring使您能够创建缓存&#xff0c;该缓存为请求范围保留数据。 例如 import org.springframework.cache.concurrent.ConcurrentMapCache; import org…

linux终止mysql进程_Ubuntu Linux下定时监测MySQL进程终止时自动重启的方法

前言最近发现MySQL服务隔三差五就会挂掉&#xff0c;导致我的网站和爬虫都无法正常运作。自己的网站是基于MySQL&#xff0c;在做爬虫存取一些资料的时候也是基于MySQL&#xff0c;数据量一大了&#xff0c;MySQL它就有点受不了了&#xff0c;时不时会崩掉&#xff0c;虽然我自…

系统错误null是什么意思_为什么NULL是错误的?

系统错误null是什么意思Java中NULL用法的简单示例&#xff1a; public Employee getByName(String name) {int id database.find(name);if (id 0) {return null;}return new Employee(id); }这种方法有什么问题&#xff1f; 它可能返回NULL而不是对象-这是错误的。 在面向对…

mysql数据库版本不同_mysql数据库版本不同所引起的问题

1.sql_mode不同所引起的问题mysql5.7 ERROR 1055 (42000): Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column lhh.lhh.id which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode…

线性搜索或顺序搜索算法在Java中如何工作? 示例教程

大家好&#xff0c;我之前谈到了二进制搜索算法的工作原理&#xff0c;并分享了在Java中实现二进制搜索的代码。 在那篇文章中&#xff0c;有人问我是否还存在其他搜索算法&#xff1f; 如果数组中的元素未排序&#xff0c;又如何使用二进制搜索算法&#xff0c;该如何搜索呢&a…