iOS打印浅析
一般设置编译时打印,发布后不打印日志
#ifdef DEBUG
#define DLog(format, ...) NSLog(format, ## __VA_ARGS__)
#else
#define DLog(format, ...)
#endif
有时候需要设置更详细的日志信息
#ifdef DEBUG
# define DLog(fmt, ...) NSLog((@"[文件名:%s]\n" "[函数名:%s]\n" "[行号:%d] \n" fmt), __FILE__, __FUNCTION__, __LINE__, ##__VA_ARGS__);
#else
# define DLog(...);
#endif
后来有遇到崩溃,需要设置编译时断言,发布后打印
#define CPAssert(condition, ...) \
if (!(condition)){ CPLog(__FILE__, __FUNCTION__, __LINE__, __VA_ARGS__);} \
NSAssert(condition, @"%@", __VA_ARGS__);
void CPLog(const char* file, const char* func, int line, NSString* fmt, ...)
{
va_list args;
va_start(args, fmt);
// NSLog(@"%s|%s|%d|%@", file, func, line, [[[NSString alloc] initWithFormat:fmt arguments:args] autorelease]);
NSLog(@"%s|%s|%d|%@", file, func, line, [[NSString alloc] initWithFormat:fmt arguments:args]);
va_end(args);
}
上面着重表现了打印的实现
可以简化成
#define DLog(fmt, ...) NSLog((@"[文件名:%s]\n" "[函数名:%s]\n" "[行号:%d] \n" fmt), __FILE__, __FUNCTION__, __LINE__, ##__VA_ARGS__);
#define CPAssert(condition, ...) \
if (!(condition)){ DLog( __VA_ARGS__);} \
NSAssert(condition, @"%@", __VA_ARGS__);
备注:断言在release时不被编译,所以调用时
CPAssert(NO,@"value is nil")
CPAssert(NO,@"[%@] is nil",anObject)
参数解释:
1.VA_ARGS是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支(VC6.0的编译器不支持)。宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的","去掉的作用,否则会编译出错, 你可以试试。
2.FILE宏在预编译时会替换成当前的源文件名
3.LINE宏在预编译时会替换成当前的行号
4.FUNCTION宏在预编译时会替换成当前的函数名称
有了以上这几个宏,特别是有了VA_ARGS,调试信息的输出就变得灵活多了。
看完这段解释,读者应该对自定义NSLog有更深的理解了。
网友评论