全局变量宏定义
发布日期:2021-06-29 01:28:00 浏览次数:2 分类:技术文章

本文共 3042 字,大约阅读时间需要 10 分钟。

 

以下是如何定义全局变量。众所周知,全局变量应该是得到内存分配且可以被其他模块通过C语言中extern关键字调用的变量。因此,必须在 .C 和 .H 文件中定义。这种重复的定义很容易导致错误。

Error[e46]: Undefined external "SerBfr" referred in ZLG522S D:\mcu\430prj\zlg522s\Debug\Obj\ZLG522S.r43 )

以下讨论的方法只需用在头文件中定义一次。虽然有点不易懂,但用户一旦掌握,使用起来却很灵活。表1.2中的定义出现在定义所有全局变量的.H头文件中。

程序清单 L 1.2 定义全局宏。

#ifdef xxx_GLOBALS

#define xxx_EXT

#else

#define xxx_EXT extern

#endif

.H 文件中每个全局变量都加上了xxx_EXT的前缀。xxx代表模块的名字。该模块的.C文件中有以下定义:

#define xxx_GLOBALS

#include "includes.h"

当编译器处理.C文件时,它强制xxx_EXT(在相应.H文件中可以找到)为空,(因为xxx_GLOBALS已经定义)。所以编译器给每个全局变量分配内存空间,而当编译器处理其他.C文件时,xxx_GLOBAL没有定义,xxx_EXT被定义为extern,这样用户就可以调用外部全局变量。

为了说明这个概念,可以参见uC/OS_II.H,其中包括以下定义:

#ifdef OS_GLOBALS

#define OS_EXT

#else

#define OS_EXT extern

#endif

 

OS_EXT INT32U       OSIdleCtr;

OS_EXT INT32U       OSIdleCtrRun;

OS_EXT INT32U       OSIdleCtrMax;

同时,uCOS_II.H有中以下定义:

#define OS_GLOBALS

#include “includes.h

当编译器处理uCOS_II.C时,它使得头文件变成如下所示,因为OS_EXT被设置为空。

INT32U       OSIdleCtr;

INT32U       OSIdleCtrRun;

INT32U       OSIdleCtrMax;

这样编译器就会将这些全局变量分配在内存中。当编译器处理其他.C文件时,头文件变成了如下的样子,因为OS_GLOBAL没有定义,所以OS_EXT被定义为extern。

extern INT32U       OSIdleCtr;

extern INT32U       OSIdleCtrRun;

extern INT32U       OSIdleCtrMax;

在这种情况下,不产生内存分配,而任何 .C文件都可以使用这些变量。这样的就只需在 .H 文件中定义一次就可以了。

 

 

【网友经验(来自CSDN)】:尽管一个全局变量或函数可以(在多个编译单元中) 有多处“声明”, 但是“定义” 却只能允许出现一次。定义是分配空间并赋初值(如果有) 的声明。最好的安排是在某个相关的.c 文件中定义, 然后在头文件(.h) 中进行外部声明, 在需要使用的时候, 只要包含对应的头文件即可。定义变量的.c 文件也应该包含该头文件, 以便编译器检查定义和声明的一致性

    这条规则提供了高度的可移植性: 它和ANSI C标准一致, 同时也兼容大多数ANSI 前的编译器和连接器。Unix 编译器和连接器通常使用“通用模式”允许多重定义, 只要保证最多对一处进行初始化就可以了; ANSI C 标准称这种行为为“公共扩展”, 没有语带双关的意思。

    如果希望让编译器检查声明的一致性, 一定要把全局声明放到头文件中。特别是, 永远不要把外部函数的原型放到.c 文件中: 通常它与定义的一致性不能得到检查, 而矛盾的原型比不用还糟糕。

 

如何定义宏、取消宏

//定义宏
#define         [MacroName]                 [MacroValue]
//取消宏
#undef         [MacroName]
普通宏
#define     PI           (3.1415926)
带参数的宏
#define     max(a,b)       ((a)> (b)?   (a),(b))
关键是十分容易产生错误,包括机器和人理解上的差异等等。所以如果用的话,建议所有的层次都要加括号。
条件编译
#ifdef   XXX…(#else)   …#endif
例如
        #ifdef   DV22_AUX_INPUT
        #define   AUX_MODE   3  
#else
        #define   AUY_MODE   3
endif
#ifndef   XXX   …   (#else)   …   #endif
头文件(.h)可以被头文件或C文件包含;
重复包含(重复定义)
由于头文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题的。
通过条件编译开关来避免重复包含(重复定义)
例如
#ifndef     __headerfileXXX__
        #
define     __headerfileXXX__
     …
        文件内容
     
#endif

#define的一些用法  

karla   发表于   2005-3-28   16:36:00  
1.简单的define定义
#define   MAX   1000  
一个简单的max就定义好了,它代表1000,如果在程序里面写
for(i=0;i{
                  ................
}
编译器在处理这个代码之前会对MAX进行处理替换为1000。
2.define的“函数定义
define可以像函数那样接受一些参数,如下
#define   max(x,y)   (x)> (y)?(x):(y);
这个定义就将返回两个数中较大的那个,看到了吗?因为这个“函数”没有类型检查,就好像一个函数模板似的,当然,它绝对没有模板那么安全就是了。可以作为一个简单的模板来使用而已。
但是这样做的话存在隐患,例子如下:
#define     Add(a,b)         a+b;
在一般使用的时候是没有问题的,但是如果遇到如:c   *   Add(a,b)   *   d的时候就会出现问题,代数式的本意是a+b然后去和c,d相乘,但是因为使用了define(它只是一个简单的替换),所以式子实际上变成了
c*a   +   b*d
另外举一个例子:
#define   pin   (int*);
pin   a,b;
本意是a和b都是int型指针,但是实际上变成
int*   a,b;
a是int型指针,而b是int型变量。
这是应该使用typedef来代替define,这样a和b就都是int型指针了。
3.宏的单行定义
#define     A(x)         T_##x
#define     B(
x)         #@x
     
#define     C(
x)         #x
      我们假设:x=1,则有:
      A(1)------〉
T_1
      B(1)------〉
'1 '
      C(1)------〉
"1 "
(这里参考了   hustli的文章)
3.define的多行定义
define可以替代多行的代码,例如MFC中的宏定义(非常的经典,虽然让人看了恶心)
  #define   MACRO(arg1,   arg2)   do   {   \
    /*   declarations   */   \
    stmt1;       \
    stmt2;       \
    /*   ...   */     \
    }   while(0)   /*   (no   trailing   ;   )   */
关键是要在每一个换行的时候加上一个 "\ "  

转载地址:https://blog.csdn.net/yu704645129/article/details/38819769 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:全局变量、extern/static/const区别与联系 (转)
下一篇:如何查看BUSHOUND内容(转)

发表评论

最新留言

不错!
[***.144.177.141]2024年04月06日 04时47分45秒