探究
看下面的一段代码
enum AudioMasterOpcodes
{
//-------------------------------------------------------------------------------------------------------
audioMasterAutomate = 0,
audioMasterVersion,
audioMasterCurrentId,
audioMasterIdle,
DECLARE_VST_DEPRECATED (audioMasterPinConnected) // 带参数宏定义
//-------------------------------------------------------------------------------------------------------
#if c
#define DECLARE_VST_DEPRECATED(identifier) __##identifier##Deprecated
#else
#define DECLARE_VST_DEPRECATED(identifier) identifier //
#endif
//調用
case audioMasterCurrentId:
return 0;
case audioMasterIdle:
case audioMasterPinConnected:
};
define 的作用
在C或C++语言源程序中允许用一个标识符来表示一个字符串,称为“宏”。
被定义为“宏”的标识符称为“宏名”。在编译预处理时,对程序中所有出现的“宏名”,都用宏定义中的字符串去代换,这称为“宏代换”或“宏展开”。宏定义是由源程序中的宏定义命令完成的。宏代换是由预处理程序自动完成的。
无参宏定义
无参宏的宏名后不带参数。
其定义的一般形式为:
#define 标识符 字符串
例如:
#define M (a+b)
它的作用是指定标识符M来代替表达式(a+b)。在编写源程序时,所有的(a+b)都可由M代替,而对源程序作编译时,将先由预处理程序进行宏代换,即用(a+b)表达式去置换所有的宏名M,然后再进行编译。
对于宏定义还要说明以下几点:
- 1. 宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。
- 2. 宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。
- 3. 宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用#undef命令。
带参宏定义
在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。
带参宏定义的一般形式为:
#define 宏名(形参表) 字符串
字符串中含有各个形参。
带参宏调用的一般形式为:
宏名(形参表)
例如:
#define M(y) ((y)*(y)+3*(y)) /*宏定义*/
单行宏定义
#define A(x) T_##x
#define B(x) #@x
#define C(x) #x
多行宏定义
关键是要在每一个换行的时候加上一个”\”
#define DORECOVERY()/
LED_RUN=1;/
VAL1=OFF;/
VAL2=OFF;/
PUMP=OFF;/
VAL3=ON;/
COMPRE=ON;
宏定义的应用场景
在大规模的开发过程中,特别是跨平台和系统的软件里,define最重要的功能是条件编译。
//就是:
#ifdef WINDOWS
......
......
#endif
#ifdef LINUX
......
......
#endif
//可以在编译的时候通过#define设置编译环境
如何定义宏、取消宏
//定义宏
#define [MacroName] [MacroValue]
//取消宏
#undef [MacroName]
普通宏
#define PI (3.1415926)
带参数的宏
#define max(a,b) ((a)>(b)? (a),(b))
关键是十分容易产生错误,包括机器和人理解上的差异等等。
防止重复定义-条件编译
头文件(.h)可以被头文件或C文件包含;重复包含(重复定义)由于头文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题的。
通过条件编译开关来避免重复包含(重复定义)
//例如
#ifndef __headerfileXXX__
#define __headerfileXXX__
…
//文件内容
…
#endif
总结
- (1)实参如果是表达式容易出问题
#define S(r) r*r
area=S(a+b);//第一步换为area=r*r;,第二步被换为area=a+b*a+b;
//正确的宏定义是
#define S(r) (r)*(r)
- (2)宏名和参数的括号间不能有空格
- (3)宏替换只作替换,不做计算,不做表达式求解
- (4)函数调用在编译后程序运行时进行,并且分配内存。宏替换在编译前进行,不分配内存
- (5)宏的哑实结合不存在类型,也没有类型转换。
- (6)函数只有一个返回值,利用宏则可以设法得到多个值
- (7)宏展开使源程序变长,函数调用不会
- (8)宏展开不占运行时间,只占编译时间,函数调用占运行时间(分配内存、保留现场、值传递、返回