由于log4cplus日志是<<形式的日志输入,然后接手的一些库是用printf()通过不定参数打印到屏幕的。想着直接将其打印到文件,便于日后查验,又不想重头到尾的修改成log4cplus格式的输入。那样要改好久,且由于不定参数的打印格式问题,估计改完后会有一堆的非期望输出。
目标将所有printf通过LOG的替换,完成将打印到屏幕转为由log4cplus接入。
一种方法是定义一个不定参数宏:
extern void LogWrite(char *s, ...);
#define LOG2(format, ...) LogWrite(format, __VA_ARGS__);
然后此不定参数宏调用一个不定参数函数,在里面实现将不定参数转为string,将后调用log4cplus日志打印出来
void LogWrite(char *format, ...)
{
va_list pArgs;
va_start(pArgs, format);
size_t nLength = vsnprintf(NULL, 0, format, pArgs) + 1;
va_end(pArgs);
char * p = NULL;
char buf[1024] = { 0 };
memset(buf, 0, sizeof(buf));
p = buf;
if (nLength >=sizeof(buf))
{
char * new_buf = new char[nLength + 1];
memset(new_buf, 0, nLength + 1);
p = new_buf;
}
va_list plist;
va_start(plist, format);
vsnprintf(p, nLength, format, plist);
va_end(plist);
string str = p;
if (nLength >= sizeof(buf))
delete[]p;
LOG_INFO(str);
}
实现也是比较简单,通过va_list获取不定参数,通过vsnprintf得到不定参数转成的str的大小,然后若发现给定的buf不够大时,new一个大一点的buf以实现转换
一切都很好,可以正常运行。只有一个问题,不管在哪调用LOG2,打印的行总在LogWrite函数的行中,定位问题的功能失去了。
因此,此功能还需要优化。
考虑不能定位到具体代码的原因是因为我们调用了具体的函数,然后在内部又调用LOG_INFO了,所以凡是LOG调用日志的地方打印出来都是在LogWrite函数的最后一行。
找到问题症结,解决方法也就出来了。直接通过宏调用的方式把前面的函数在宏中实现:
# include <stdarg.h>
# include <string.h>
# include <ctype.h>
# include <time.h>
# include <memory>
#define LOG(format, ...) \
{\
char buff[1024] = { 0 };\
memset(buff, 0, sizeof(buff));\
int maxLength = _snprintf(NULL, 0, format, __VA_ARGS__) + 1;\
int nLength = maxLength < sizeof(buff) - 1 ? maxLength : sizeof(buff) - 1;\
_snprintf(buff, nLength, format, __VA_ARGS__); \
LOG_INFO(buff);\
}
这个宏针对过多的日志截断为sizeof(buf)-1的长度,由于是宏,因此需要拿宏的不定参数__VA_ARGS__的方式。原先的vsnprintf由于参数不匹配需弃用,改用_snprintf。
前面遇到的最大的坑即为在函数中获取不定参数用的是va_list pArgs,而在宏中则是直接调用__VA_ARGS__,我在这一块反复试了好久,前后几个小时才算通关。
这样完美实现前述需求。