define常规的文字替换就不多说明了,
先说一下带参数的宏替换,比如
#define add(x, y) (x + y)
需要注意的就是在涉及运算或着其他一些情况下,要加上括号来避免结合律影响运算结果,像5*add(2,3),你期望的结果是25,但是,在不加括号的情况下 5*2+3 结果是30.
当需要换行时,需要在行尾加上 比如:
1 #define NULL_RETURN(varName) 2 if(varName == nullptr) 3 { 4 return; 5 }
下面说下宏定义中的特殊操作符,包括#,##和__VA_ARGS__
#符号说白了就是可以把一个参数当作字符串来处理,避免参数替换,比如:
#define SQR(x) printf("the square of" #x "is %d.
",(x)*(x))
当我们不希望printf中的x被参数替换时可以加上#号。
##符号简单理解就是将两个字符串拼接成一个,比如
#define PROPERTY_READONLY(varType, funName, varName) private: varType _##varName; public: varType get##funName() const { return _##varName; }
__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏__VA_ARGS__就可以被用在替换部分中,替换省略号所代表的字符串。
#define PR(...) printf(__VA_ARGS__) int main() { int wt=1,sp=2; PR("hello "); //输出:hello PR("weight = %d, shipping = %d",wt,sp); //输出:weight = 1, shipping = 2 return 0; }
省略号只能代替最后面的宏参数。
#define W(x,…,y)错误!
但是支持#define W(x, …),此时传入的参数个数必须能够匹配。
介绍几个系统宏:
1) __FILE__ 预编译时会替换成当前的源文件名
2) __LINE__ 预编译时会替换成当前的行号
3) __FUNCTION__ 预编译时会替换成当前的函数名称
如何取消定义的宏呢,可以这样做
//定义宏 #define [MacroName] [MacroValue] //取消宏 #undef [MacroName]