开始阅读log4cplus源代码,这个库规模不大,但作者的C++功底不错(至少我个人认为),而且更新的相当频繁,作为学习C++的实际分析对象不错。log4cplus源代码中用到了Singleton、Factory Method、Bridge、Observer等等Pattern,还有Pimpl机制、引用计数机制等等,非常值得总结学习。不过,这篇文章,主要用来记录,阅读代码过程中的C++小知识点,偶有所得,都会持续更新,也希望得到有缘进来的朋友的留言指正,先谢啦。
2012.10.9记录
源代码中大量出现这样的宏定义
#ifdef UNICODE
# define LOG4CPLUS_TEXT2(STRING) L##STRING
#else
# define LOG4CPLUS_TEXT2(STRING) STRING
#endif // UNICODE
#define LOG4CPLUS_TEXT(STRING) LOG4CPLUS_TEXT2(STRING)
理解起来并不难,主要是为了Unicode下,字符串前需要L,转换为WCHAR。在宏定义中可以用##运算符把前后两个预处理Token连接成一个预处理Token。假如我们define了UNICODE这个宏,那么LOG4CPLUS_TEXT("Second"),也就相当于L
"Second";否则,就等于"Second"。问题是,为什么要多加一层宏定义,直接像下面不可以吗?
#define LOG4CPLUS_TEXT(STRING) L##STRING
原因就在于,凡宏定义里有用'#'或'##'的地方宏参数是不会再次展开的。假如我们没有使用作者的代码中的中间转换宏,遇到类似下面这样的宏展开就会得到不是期望的结果。
#define LOG4CPLUS_TEXT(STRING) L##STRING
#define MYSTRING "MYLOG"
当我们遇到
LOG4CPLUS_TEXT(
MYSTRING),
就会展开成
LMYSTRING,而不是我们期望的L"MYLOG"。加一个中间转换宏,就是为了要让宏参数完全展开。