0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

模块化原理和方法 模块化的方法和注意事项

STM32嵌入式开发 来源:STM32嵌入式开发 2023-07-25 16:43 次阅读

引言

当项目小组做一个相对较复杂的工程时,意味着你不再独自单干。而是和小组成员分工合作,这就要求小组成员各自负责一部分工程。比如你可能只是负责通讯或者显示这一块。

这个时候,你就应该将自己的这一块程序写成一个模块,单独调试,留出接口供其它模块调用。

最后,小组成员都将自己负责的模块写完并调试无误后,由项目组长进行组合调试。

像这些场合就要求程序必须模块化。模块化的好处是很多的,不仅仅是便于分工,它还有助于程序的调试,有利于程序结构的划分,还能增加程序的可读性和可移植性。

初学者往往搞不懂如何模块化编程,其实它是简单易学,而且又是组织良好程序结构行之有效的方法之一。

本文将先大概讲一下模块化的方法和注意事项,最后将以初学者使用最广的keil c编译器为例,给出模块化编程的详细步骤。

模块化程序设计应该理解以下概述:

模块即是一个.c 文件和一个.h 文件的结合,头文件(.h)中是对于该模块接口的声明;

这一条概括了模块化的实现方法和实质:将一个功能模块的代码单独编写成一个.c文件,然后把该模块的接口函数放在.h文件中.举例:假如你用到液晶显示,那么你可能会写一个液晶驱动模块,以实现字符、汉字和图像的现实,命名为: led_device.c,该模块的.c文件大体可以写成:

1af6b2a0-2ac2-11ee-a368-dac502259ad0.png     注:此处只写出这两个函数,第一个延时函数的作用范围是模块内,第二个,它是其它模块需要的。为了简化,此处并没有写出函数体.     .h文件中给出模块的接口.在上面的例子中, 向LCD写入字符函数:wr_lcd (uchar dat_comm,uchar content)就是一个接口函数,因为其它模块会调用它,那么.h文件中就必须将这个函数声明为外部函数(使用extrun关键字修饰),另一个延时函数:void delay (uint us)只是在本模块中使用(本地函数,用static关键字修饰),因此它是不需要放到.h文件中的。     .h文件格式如下: 1b12391c-2ac2-11ee-a368-dac502259ad0.png     这里注意三点:

在keil 编译器中,extern这个关键字即使不声明,编译器也不会报错,且程序运行良好,但不保证使用其它编译器也如此。强烈建议加上,养成良好的编程规范。

.c文件中的函数只有其它模块使用时才会出现在.h文件中,像本地延时函数static void delay (uint us)即使出现在.h文件中也是在做无用功,因为其它模块根本不去调用它,实际上也调用不了它(static关键字的限制作用)。

注意本句最后一定要加分号”;”,相信有不少同学遇到过这个奇怪的编译器报错: error C132: 'xxxx': not in formal parameter list,这个错误其实是.h的函数声明的最后少了分号的缘故。

模块的应用:假如需要在LCD菜单模块lcd_menu.c中使用液晶驱动模块lcd_device.c中的函数void wr_lcd (uchar dat_comm,uchar content),只需在LCD菜单模块的lcd_menu.c文件中加入液晶驱动模块的头文件lcd_device.h即可。 1b31db0a-2ac2-11ee-a368-dac502259ad0.png

某模块提供给其它模块调用的外部函数及数据需在.h 中文件中冠以extern 关键字声明;

这句话在上面的例子中已经有体现,即某模块提供给其它模块调用的外部函数和全局变量需在.h 中文件中冠以extern 关键字声明。 下面重点说一下全局变量的使用。使用模块化编程的一个难点(相对于新手)就是全局变量的设定,初学者往往很难想通模块与模块公用的变量是如何实现的,常规的做法就是本句提到的,在.h文件中外部数据冠以extern关键字声明。 比如上例的变量value就是一个全局变量,若是某个模块也使用这个变量,则和使用外部函数一样,只需在使用的模块.c文件中包含#include“lcd_device.h”即可。 另一种处理模块间全局变量的方法来自于嵌入式操作系统uCOS-II,这个操作系统处理全局变量的方法比较特殊,也比较难以理解,但学会之后妙用无穷,这个方法只需用在头文件中定义一次。方法为: 在定义所有全局变量(uCOS-II将所有全局变量定义在一个.h文件内)的.h头文件中: 1b5cd72e-2ac2-11ee-a368-dac502259ad0.png     .H 文件中每个全局变量都加上了xxx_EXT的前缀。xxx 代表模块的名字。     该模块的.C文件中有以下定义: 1b6f4544-2ac2-11ee-a368-dac502259ad0.png     当编译器处理.C文件时,它强制xxx_EXT(在相应.H文件中可以找到)为空,(因为xxx_GLOBALS已经定义)。     所以编译器给每个全局变量分配内存空间,而当编译器处理其他.C 文件时,xxx_GLOBAL没有定义,xxx_EXT 被定义为extern,这样用户就可以调用外部全局变量。为了说明这个概念,可以参见uC/OS_II.H,其中包括以下定义: 1b862eda-2ac2-11ee-a368-dac502259ad0.png     同时,uCOS_II.H 中有以下定义: 1b97ef26-2ac2-11ee-a368-dac502259ad0.png     当编译器处理uCOS_II.C 时,它使得头文件变成如下所示,因为OS_EXT 被设置为空。 1ba9cb10-2ac2-11ee-a368-dac502259ad0.png     这样编译器就会将这些全局变量分配在内存中。当编译器处理其他.C 文件时,头文件变成了如下的样子,因为OS_GLOBAL没有定义,所以OS_EXT 被定义为extern。 1bbebe58-2ac2-11ee-a368-dac502259ad0.png     在这种情况下,不产生内存分配,而任何 .C文件都可以使用这些变量。这样的就只需在 .H文件中定义一次就可以了。

模块内的函数和全局变量需在.c 文件开头冠以static 关键字声明;

这句话主要讲述了关键字static的作用。Static是一个相当重要的关键字,他能对函数和变量做一些约束,而且可以传递一些信息。 比如上例在LCD驱动模块.c文件中定义的延时函数static void delay (uint us),这个函数冠以static修饰,一方面是限定了函数的作用范围只是在本模块中起作用,另一方面也给人传达这样的信息:该函数不会被其他模块调用。 下面详细说一下这个关键字的作用,在C 语言中,关键字static 有三个明显的作用:

在函数体,一个被声明为静态的变量在这一函数被调用过程中维持其值不变。

在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问。它是一个本地的全局变量。

在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那就是,这个函数被限制在声明它的模块的本地范围内使用。

前两个都比较容易理解,最后一个作用就是刚刚举例中提到的延时函数(static void delay (uint us)),本地化函数是有相当好的作用的。

永远不要在.h 文件中定义变量!

比较一下代码: 代码一: 1bd0161c-2ac2-11ee-a368-dac502259ad0.png     以上程序的结果是在模块1、2、3 中都定义了整型变量a,a 在不同的模块中对应不同的地址元,这个世界上从来不需要这样的程序。正确的做法是:     代码二: 1be016d4-2ac2-11ee-a368-dac502259ad0.png     这样如果模块1、2、3 操作a 的话,对应的是同一片内存单元。       注:一个嵌入式系统通常包括两类(注意是两类,不是两个)模块:

硬件驱动模块,一种特定硬件对应一个模块;

软件功能模块,其模块的划分应满足低偶合、高内聚的要求。

责任编辑:彭菁

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编程
    +关注

    关注

    88

    文章

    3637

    浏览量

    93924
  • 模块化
    +关注

    关注

    0

    文章

    333

    浏览量

    21426
  • 编译器
    +关注

    关注

    1

    文章

    1642

    浏览量

    49246
收藏 人收藏

    评论

    相关推荐

    模块化UPS电源设备的配置

    模块化UPS电源是重要的电气设备,为高效发挥其功能,从模块化UPS电源设备的主要组成部分——主机柜、强电柜、蓄电池、电池开关柜、电池架、连接线缆等方面,分析了合理配置模块化UPS电源的方法
    发表于 11-01 14:36 2117次阅读

    精粹的c语言,模块化的c编程方法

    精粹的c语言,模块化的c编程方法希望本资料能够有所对你助力。
    发表于 10-11 14:50

    单片机的C语言模块化编程

    详细介绍了主函数、头文件等,详细介绍了在KEIL软件下的编程步骤,解读了单片机模块化编程的步骤,以及注意事项
    发表于 06-13 16:21

    嵌入式系统模块化设计有什么方法

    嵌入式系统设计要求做到可测性、高效性和灵活性。目前,嵌入式系统物理尺寸越来越小,功能越来越复杂。为了方便调试、维护系统,完全可测显得极为重要。另一方面,模块化的设计方法越来越引起人们的关注。模块化
    发表于 08-23 07:31

    到底什么是模块化编程?

    对于一些接触单片机编程不久的小萌新来说,模块化编程这个概念刚接触的时候可能会很懵,到底什么是模块化编程?我以前也不懂,后面根据网上的说明和自己对库函数例程的分析,已经完全掌握了这门技能。 模块化编程
    发表于 12-03 07:39

    什么是模块化编程?模块化编程的注意事项

    单片机零基础入门(8-4)模块化编程---LED1602调试工具一、回顾二、什么是模块化编程?1、传统方式编程:2、模块化编程3、模块化编程框图3、
    发表于 02-23 07:14

    模块化程序设计简单解释

    方法。在单片机的简单解释:把各个模块的代码放在不同的.c文件里,在.h文件里提供外部可调用函数的声明,其它.c文件想使用其中的代码时,只需要#include "XXX.h"文件即可。使用模块化编程可极大的提高代码的可阅读性、可维
    发表于 02-23 06:05

    模块化Java:动态模块化

    在前一篇文章《模块化Java:静态模块化》中,我们讨论了如何构建Java模 块并将其作为一个单独的JAR进行部署。文中的例子给出了一个client和一个 server bundle(两者在同一个VM中)
    发表于 12-01 11:54 24次下载

    串行RapidIO提升模块化基站设计

    串行RapidIO提升模块化基站设计 蜂窝基站的模块化设计和制造对组合视频、语音和数据等 3G 移动服务,即通常所说的“三重服务”至关重要。但是,为什么模块化设计
    发表于 03-10 13:56 1113次阅读
    串行RapidIO提升<b class='flag-5'>模块化</b>基站设计

    基于模块化设计方法实现FPGA动态部分重构

      动态部分重构可以通过两种方法实现:基于模块化设计方法(Module-Based ParTIal Reconfiguration)和基于差别的设计方法(Difference-Base
    发表于 08-23 10:35 752次阅读
    基于<b class='flag-5'>模块化</b>设计<b class='flag-5'>方法</b>实现FPGA动态部分重构

    012-IIC总线原理和模块化编程方法

    云龙单片机—IIC总线原理和模块化编程方法
    发表于 03-17 11:03 7次下载

    模块化程序设计

    模块化程序设计思想,单片机c语言的模块化设计,方便移植,将程序封装备用。
    发表于 03-22 15:29 9次下载

    模块化方法注意事项

    这一条概括了模块化的实现方法和实质:将一个功能模块的代码单独编写成一个.c文件,然后把该模块的接口函数放在.h文件中.举例:假如你用到液晶显示,那么你可能会写一个液晶驱动
    的头像 发表于 09-20 10:37 3229次阅读

    什么是模块化自动

    什么是模块化自动
    的头像 发表于 03-10 16:29 2923次阅读
    什么是<b class='flag-5'>模块化</b>自动<b class='flag-5'>化</b>?

    模块化插座接线方法有哪些

    扩展或改变插座的功能。以下是一些模块化插座接线方法的概述,以及一些安全和安装的注意事项。 1. 基本接线方法 模块化插座的基本接线
    的头像 发表于 10-18 09:50 749次阅读