最近在android的某个代码的头文件中发现很多__VA_ARGS__,google一下.还是比较有用.附带其它宏定义参数,一起记录之.
1. ...和__VA_ARGS__
看看msdn上给得例子吧
#include
#define EMPTY
#define CHECK1(x, ...) if (!(x)) { printf(__VA_ARGS__); }
#define CHECK2(x, ...) if ((x)) { printf(__VA_ARGS__); }
#define CHECK3(...) { printf(__VA_ARGS__); }
#define MACRO(s, ...) printf(s, __VA_ARGS__)
int main() {
CHECK1(0, "here %s %s %s", "are", "some", "varargs1(1)\n");
CHECK1(1, "here %s %s %s", "are", "some", "varargs1(2)\n"); // won't print
CHECK2(0, "here %s %s %s", "are", "some", "varargs2(3)\n"); // won't print
CHECK2(1, "here %s %s %s", "are", "some", "varargs2(4)\n");
// always invokes printf in the macro
CHECK3("here %s %s %s", "are", "some", "varargs3(5)\n");
MACRO("hello, world\n");
// MACRO("error\n", EMPTY); would cause C2059
}
here are some varargs1(1)
here are some varargs2(4)
here are some varargs3(5)
hello, world
是不是很有趣,以后我们如果在一些情况下要定义自己的打印函数就特别方便了
比如我们要在A条件成立,且B条件不成立的情况下打印输出
可以做如下定义:
#define CHECK1(A,B,...) if ((A)&&!(B)) { printf(__VA_ARGS__); }
#define W(x,...,y)
2. #
这个特殊的宏定义参数也特别有用.#作为一个预处理运算符,它可以把语言符号字符串化(stringizing).例如我们定义的变量等.
看一个简单的例子:
#include
#define TEST(x) printf("square of " #x " is %d.\n",(x)*(x))
void main()
{
int y =4;
TEST(y);
TEST(6-3);
TEST(y+3);
}
输出结果:
square of y is 16.
square of 6-3 is 9.
square of y+3 is 49.
##运算符可以用于类函数宏的替换部分.##还可以用于类对象宏的替换部分.这个运算符可以把两个语言符号组合成单个语言符号.
例如:
守护进程则可以改为syslog等等... 其中,决窍其实就是这几个宏 ##__VA_ARGS__, __FILE__, __LINE__ 和__FUNCTION__,下面介绍一下这几个宏:
1) __VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的","去掉的作用,否则会编译出错, 你可以试试。
2) __FILE__ 宏在预编译时会替换成当前的源文件名
3) __LINE__宏在预编译时会替换成当前的行号
4) __FUNCTION__宏在预编译时会替换成当前的函数名称
3.可变参数宏 ...和_ _VA_ARGS_ _
其中,...表示参数可变, __VA_ARGS__定义好的宏
__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串。比如:
#define PR(...) printf(__VA_ARGS__)
int main()
{
int wt=1,sp=2;
PR("hello\n");
PR("weight = %d, shipping = %d",wt,sp);
return 0;
}
省略号只能代替最后面的宏参数。
#define W(x,...,y)错误!
飞行棋例子:
#define CHARFMT(__format__,...) String::createWithFormat(__format__,##__VA_ARGS__)->getCString()
#define DYNAMIC_CAST_READER(__type__,__path__) dynamic_cast<__type__*>(cocostudio::GUIReader::getInstance()->widgetFromJsonFile(__path__));
#define OBJ2OBJ_STATIC_CAST(__ccobject__,__type__) static_cast<__type__*>(__ccobject__)
#define OBJ2OBJ_DYNAMIC_CAST(__ccobject__,__type__) dynamic_cast<__type__*>(__ccobject__)
#define AUDIO_PLAYBACKGROUND(__format__) MusicManager::getInstance()->playBackgroundMusic(__format__);
#define AUDIO_PLAYEFFECT(__format__) MusicManager::getInstance()->playEffectMusic(__format__);
4 常量及宏的命名
采用下划线分割大写字母的方式命名,一般应以设备名作为前缀,
防止模块间命名的重复。如:
#define TIMER0_MODE_RELOAD 2
#define TIMER2_COUNT_RETRIEVE(val) ((uint16)(65536 - (val)))
当然,看作接口的宏可以按照函数的命名方法命名,例如:
#define timer2_clear() (TF2 = 0)
#define timer0_is_expired() (TF0)
----------------------
#define CHARFMT(__format__,...) String::createWithFormat(__format__,##__VA_ARGS__)->getCString()
这句话使用下面更具有健状性
#define CONVERT_TO_INT(__format__ , ...)\
do {\
if (__format__)\
{\
CCLOG(__VA_ARGS__);\
}\
} while (0);
CONVERT_TO_INT(10, "herr is %s %s", "hello", "are");
注: 1 第一个 ##不能去掉 ? 第二个为何不用加 ##
2 测试发现 第一个参数__format__ 多余 weisha?
3 名字一定要大写
4 \ 之前不能有空格 老王
#define XNAME(n) x##n
执行宏调用:
int XNAME(4)
展开后:
x4;//相当于直接定义 int x4 这里也就体现了两个语言符号的拼接.不过在android经常看到有关##拼接语言符号的情况.