很多时候我们都希望一些函数能够支持可变参数,这样的函数或是用来进行字符串的格式化,或是调用函数时函数本身支持参数的格式化(这样在调用函数之前不用另外格式化,调用的函数本身就支持参数的格式化)。比如C语言中的printf函数,MFC中的CString的Format函数。它们内部均是使用va_start、va_list、va_end实现参数的解析的。下面给出duilib中的CStdString::Format 的函数实现,简单的查看一下是如何解析的。
int CStdString::Format( LPCTSTR pstrFormat, ... )
{
CStdString sFormat = pstrFormat;
// Do ordinary printf replacements
// NOTE: Documented max-length of _vstprintf() is 1024
TCHAR szBuffer[1025] = { 0 };
va_list argList;
va_start( argList, pstrFormat );
// wvsprintf不支持浮点格式,所以换成_vstprintf
int iRet = ::_vstprintf( szBuffer, sFormat, argList );
va_end( argList );
Assign( szBuffer );
return iRet;
}
上面简要的说明了使用va_start、va_list、va_end实现了对可变参数格式化的支持,那其中的陷阱在什么地方呢?在日常的代码开发过程中,为了定位问题我们需要添加相关的打印信息,或打印到控制台窗口上,或是将日志写入到文件中。我们就以C语言中打印输出函数printf为例吧。该函数