#ifdef DEBUG
#define DebugLog( s, ... ) printf( "<%s:(%d)> %s: %s\n\n", [[[NSString stringWithUTF8String:__FILE__] lastPathComponent] cStringUsingEncoding:NSUTF8StringEncoding], __LINE__,__func__, [[NSString stringWithFormat:(s), ##__VA_ARGS__] cStringUsingEncoding:NSUTF8StringEncoding])
#else
#define DebugLog( s, ... )
#endif
此处 DebugLog( s, ... )的作用是 把它替换成 “printf( "<%s:(%d)> %s: %s\n\n", [[[NSString stringWithUTF8String:__FILE__] lastPathComponent] cStringUsingEncoding:NSUTF8StringEncoding], __LINE__,__func__, [[NSString stringWithFormat:(s), ##__VA_ARGS__] cStringUsingEncoding:NSUTF8StringEncoding])”
1. __VA_ARGS__:总体来说就是将左边宏中"( s, ... )" 的内容原样抄写在右边 __VA_ARGS__ 所在的位置,它是一个可变参数的宏。要注意的是,printf 的输出格式是括号内左边是字符串,右边是变量,而且右变量与左输出格式是一一对应的,用法和NSLog差不多。
宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的","去掉的作用,否则会编译出错, 你可以试试。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串
2. __FILE__ :宏在预编译时会替换成当前的源文件名
3. __LINE__:宏在预编译时会替换成当前的行号
4. __FUNCTION__:宏在预编译时会替换成当前的函数名称
5. __TIME__:宏在预编译时会替换成系统的当前时间
6.__TIMESTAMP__:和第五点类似
7.##运算符可以用于类函数宏的替换部分。另外,##还可以用于类对象宏的替换部分。这个运算符把两个语言符号组合成单个语言符号
例:
DebugLog(@"123");
输出:
<ViewController.m:(34)> 11:40:14Tue Jan 12 11:11:16 2016 -[ViewController viewDidLoad]: 123
了解更多:http://www.cnblogs.com/pengyingh/articles/2407265.html