当前位置:首页 > 公众号精选 > 玩转嵌入式
[导读]随着我们工程化经验的增加,不知不觉的我们就会关心到这个问题,模块化,模块设计就显现出来,那到底什么是模块化呢?


随着我们工程化经验的增加,不知不觉的我们就会关心到这个问题,模块化,模块设计就显现出来,那到底什么是模块化呢?

这不叫模块化

我相信在很多时候,我们刚开始从零开始接手一个项目的时候,编码之前总想着要实现什么的功能需要的模块,然后要程序 模块化,这种思想是值得认同的,但往往我们并没有做到真正的模块化。 例如在一个需要很多外设接口,一般需要硬件初始化,相关配置,中断服务程序,输入输出,逻辑处理等功能,我们的做法可能就是把代码分布到多个文件和目录里面,然后把这些目录或者文件取名 xxxModule 甚至把这些目录分放在不同的仓库目录里面,结果随着编码的增加,发现好多小功能都重复了,或者本可以写在一起的函数并没有放在一起,导致我们的代码思想不是很流畅,这样做会误导我们,甚至整个项目实现的思路。 究其原因这是因为我们其实并不理解什么叫做 模块,而仅仅是肤浅的把代码切割开来,分放在不同的位置,虽然这确实达到了部分模块化的目的,但是也会制造一些不必要的麻烦。

什么是真正的模块化?

真正的模块化,并不是简单文本意义上的,而是与逻辑相关的有逻辑意义的。一个模块应该像一个集成电路芯片,我们能见到能使用的都很清晰,它定义了良好的输入和输出。 模块是可能分开地被编写的单位。这使他们可再用和允许广泛人员同时协作、编写及研究不同的模块。 实际上,编程语言已经为我们提供了一种很好的模块化方法,它的名字叫做 函数。每一个函数都有明确的 输入(参数)和输出(返回值),同一个文件里可以包含多个函数,所以你其实根本不需要把代码分开在多个文件或者目录里面,同样可以完成代码的模块化。 按照函数这个原则,我可以把代码全都写在同一个文件里,却仍然是非常模块化的代码,是不是觉得与之前的想法不一样? 是的,软件编程模块是一套一致而互相有紧密关联的软件组织, 每个模块完成一个特定的子功能,所有的模块按某种方法组装起来,成为一个整体,完成整个系统所要求的功能,这就是真正的模块化。

怎么模块化?

我们知道了模块化的原则与道理之后,就可以按照这个思路去开发项目了,想要达到很好的模块化,你需要做到以下几点。我们从实现角度来说。

避免写太长的函数

一眼望去,如果一个函数的代码你电脑一页都看不完,那肯定是冗长了或者不符合模块化编程了。不需要滚屏就可以看得到全部的函数内容,那对我们的理解也有帮助。 一般来说一个函数最好不要超过40行代码(当然这个不是死规定,只是一个经验建议而已),如果写的函数太大了,就应该把它拆分成几个更小的函数。 也许你会说到,这很难办到,逻辑很多或者很多判断条件的时候,40行往往不够吧,那么其实我们也需要考虑到函数里面一些复杂的部分,是不是可以提取出来,单独写一个小函数,再从原来的函数里面调用。 另外函数层级也不要太多,比如一个函数长成这样:
      1void function(void* para)  2{  3 if (getOS().equals("MacOS")) {  4 a();  5 if(getOS().equals("AndroidOS")){  6 b();  7 if(getOS().equals("flymeOS")){  8 c();  9 } 10 } 11 } 12}  
我们看到这个函数由于很多的判断,函数层级已经超过4层了,这其实对我们的理解很不利,另一方面,一些逻辑变化也会导致我们的更改很麻烦,费脑子。

每个函数只做一件简单的事情

有些人喜欢写一些 通用函数,一般我都放在 publicModule里面,既可以实现这个工又可以实现那个功能,它的内部依据某些变量或者是某些条件,来选择这个函数所要实现的小功能。 比如,写出这样的函数:
      1void function() {  2 if (getOS().equals("MacOS")) {  3 a();  4 } else {  5 b();  6 }  7  8 c();  9 10 if (getOS().equals("MacOS")) { 11 d(); 12 } else { 13 e(); 14 } 15}  
这个函数,是想表达根据系统是否为MacOS,从而来做不同的事情。从这个函数可以很容易的看出,其实只有函数c()是两种系统共有的,而其它的函数 a(), b(), d(), e()都属于不同的分支。 但是这种复用的写法其实是很不利的。如果一个函数可能实现2个功能,并且它们之间 共同点少于它们的不同点,那我们最好就写两个不同的函数,否则这个函数的逻辑就不会很清晰,容易出现错误。 不要害怕,函数简单点不丢人,我们不需要炫技。 好了,根据上面的说法,这个函数可以改写成两个函数:
     1void funMacOS() { 2 a(); 3 c(); 4 d(); 5}  
     1void funOther() { 2 b(); 3 c(); 4 e(); 5}  
如果我们发现两件事情大部分内容相同,只有少数不同,也就是说 共同点大于它们的不同点,那就更简单了,我们可以把相同的部分提取出去,做成一个 辅助函数 比如,如果有个函数是这样:
      1void function() {  2  3 a();  4 b()  5 c();  6  7 if (getOS().equals("MacOS")) {  8 d();  9 } else { 10 e(); 11 } 12}  
其中函数 a(),b(),c()都是一样的,只有函数 d()、e()根据系统有所不同。那么你可以把函数 a(),b(),c()提取出去,代码如下:
     1void preFun() { 2 a(); 3 b() 4 c(); 5}  
然后分别写两个函数:
     1void funMacOS() { 2 preFun(); 3 d(); 4}  
     1void funOther() { 2 preFun(); 3 e(); 4}  
这样更改之后,是不是清晰了很多,我们既共享了代码,避免冗余,又做到了每个函数只做一件简单的事情。这样的代码,逻辑就更加清晰了。

工具函数

是的,我再说一遍,每个函数只做一件简单的事情。但是有些时候在我们的工程代码中,我们可能会发现其实里面有很多的重复。 所以一些常用的代码,不管它有多短或者多么的简单,都可以提取出去做成函数,例如有些帮助函数也许就只有2行,但是我们把它封装成一个函数的话,就能大大简化主要函数里面的逻辑。 也许你可能会说,这些函数调用会增加代码开销,但随着硬件发展以及技术变革,这已经是一种过时的观念了。 现代的很多编译器都能自动的把小的函数 内联(inline)到调用它的地方,所以根本不会产生函数调用,也就不会产生任何多余的开销。 那么我可以使用宏来代替工具函数?一行代码搞定了,比如
     1#define FillAndSendTxOptions( TRANSSEQ, ADDR, ID, LEN, TxO ) { \ 2afStatus_t stat;                                    \ 3ZDP_TxOptions = (TxO);                              \ 4stat = fillAndSend( (TRANSSEQ), (ADDR), (ID), (LEN) );          \ 5ZDP_TxOptions = AF_TX_OPTIONS_NONE;                 \ 6return stat;                                        \ 7}  
同样,这也许也过时了,我不想让宏(macro)来背这个锅,在早期的C语言编译器里,只有宏是静态内联的,所以使用宏是为了达到内联的目的。 然而能否内联,其实并不是宏与函数的根本区别,这里我不细说了,只要记住: 应该尽量避免使用宏。如果想了解可以参考 避免这7个误区,才能让【宏】削铁如泥 为了内联而使用宏,其实是滥用了宏,这会引起各种各样的麻烦,比如使程序难以理解,难以调试,容易出错等等。

尽量使用局部变量和参数

我们应该尽量避免使用全局变量和类成员(class member)来传递信息,举个例子:
      1class A {  2 String x;  3  4 void findX() {  5 ...  6 x = ...;  7 }  8  9 void fun() { 10 findX(); 11 ... 12 print(x); 13 } 14}  
首先,使用函数findX(),把一个值写入成员x。然后,调用x的值。这样,x就变成了findX和print之间的数据通道。 由于 x属于class A,这样程序就失去了模块化的结构,与我们所说的模块化意义不符了。两个函数依赖于成员x,就不再有明确的输入和输出,而是依赖全局的数据。 函数findX和fun不再能够离开 class A而存在,具有依赖性,并且由于类成员还有可能被其他代码改变,这样就会导致代码变得复杂难以理解,函数的正确性也难以保证。 如果使用局部变量和参数来传递信息,那么这两个函数就不需要依赖于某一个class,不易出错,代码如下:
     1String findX() { 2 ... 3 x = ...; 4 return x; 5 } 6 void foo() { 7 String x = findX(); 8 print(x); 9 }  

总结

模块化是指解决一个复杂问题时,自顶向下,逐层把系统划分成若干模块的过程,深入理解模块化,什么是真正的模块化,那么我们才能够事半功倍。 如果你喜欢我的文章表达的思维,转发分享是对我的厚爱,期待您的点赞在看,下一期,再见!


	

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

本文将详细指导读者如何制作一个限位断路开关,包括所需材料、制作步骤、注意事项等。通过DIY限位断路开关,读者可以更好地理解限位断路开关的工作原理,同时提高动手能力和创新意识。

关键字: 限位断路开关 电气控制元件 模块化

虽然模块化数据中心为行业带来了令人兴奋的可能性,但它并不是一种万能的解决方案。那么,如何合理的模块化使用呢?它何时有效,何时无效?

关键字: 模块化 数据中心

2023年10月18日,中国在第三届“一带一路”国际合作高峰论坛期间发布《全球人工智能治理倡议》,围绕人工智能发展、安全、治理三方面系统阐述了人工智能治理中国方案。

关键字: 人工智能 大模型 代码

我们看到这么多的安全问题,部分原因在于我们对待安全的方式:安全性通常被认为是事后考虑的问题,是在开发结束时才添加到设备上的东西。然而,复杂的系统,尤其是嵌入式系统,有一个很大的攻击面,这让攻击者有机可乘,能够在“盔甲”上...

关键字: 代码 嵌入式系统 软件漏洞

近日,第11届EEVIA年度中国硬科技媒体论坛暨产业链研创趋势展望研讨会在深圳召开。ADI公司亚太区电源市场经理黄庆义在会上发布了主题为“泛在的高性能电源技术和解决方案 正在如何演进”的演讲。

关键字: ADI 电源 Silent Switcher 模块化

新富人群财务需求多元发展,投顾服务迎来新机遇 上海2023年9月20日 /美通社/ -- 2023年9月19日,上海交通大学上海高级金融学院(高金)与全球领先的金融服务机构嘉信理财(Charles Schwab)联合发...

关键字: BSP ADVANCED INA 代码

北京2023年9月14日 /美通社/ -- 生物医药高科技公司诺诚健华(港交所代码:09969;上交所代码:688428)今日宣布,新型蛋白酪氨酸磷酸酶SHP2变构抑制剂ICP-189联用针对表皮生长因子受体(EGFR)...

关键字: IC HP 代码 ARMA

上海2023年9月1日 /美通社/ -- 2023上半年,安集科技(股票代码:688019)市场拓展规划成效显现,营业收入稳健增长。 全球半导体产业挑战持续存在的情形下,安集科技秉承发扬"克难攻坚,敢打硬...

关键字: 安集科技 BSP 代码 半导体材料

迎接云端与边缘计算融合时代 上海2023年8月29日 /美通社/ -- 如今,便携消费电子设备不仅直接支持个人数据生活,还引领着全球网络数据的爆炸式增长。根据IDC发布的《数据时代2025》报告,自2018年的33ZB...

关键字: 模块化 电子 边缘计算 人工智能

国际酒店运营商升级其在线支付功能 上海2023年8月28日 /美通社/ -- 加拿大金融科技公司Nuvei Corporation(以下简称“Nuvei”或“公司”)(纳斯达克代码:NVEI)(多伦多证券交易所代码:N...

关键字: 代码 IP SE 纳斯达克
关闭
关闭