以前一直好奇可变参数究竟是如何实现的,怎么传了几个...就能解析出真实需要的东西 呢,刚好项目有个场合需要用到这个东西,就了解了下可变参数的实现,同时感谢加俊的指导。
其实可变参数主要依赖几个重要的api,外表看起来很神奇,但其实有了API,就是很简单的事情。主要有如下几个API:
va_start
va_arg
va_end
有了这三个API以后,就能编写简单的可变参数函数啦,先贴个我第一次写的一个可变参数打印函数,有点挫:
void PrintChars(const char* szMgs,...)
{
std::string strMsg(szMgs);
size_t Pos =0;
size_t BeginPos=0;
std::vector<std::string> vtring;
size_t iPos = 0;
int val=0;
va_list vl; // 用来存放取出的参数
va_start(vl,szMgs); //
char buf[33] = {0};
while ((Pos = strMsg.find("%",BeginPos)) != std::string::npos) // 我只是为了找到百分号
{
std::string stringTmp(strMsg.c_str()+BeginPos,Pos-BeginPos);
val = va_arg(vl,int); // 这个就是取出具体的那个可变参数的值啦,那个int,是只那个可变参数的数据类型,所以严谨的来说,前面需要判断%后面那个是否是d,
stringTmp= stringTmp + itoa(val,buf,10);
vtring.push_back(stringTmp);
BeginPos=Pos+2;
}
// 结束以后再做一次
{
std::string stringTmp(strMsg.c_str()+BeginPos);
vtring.push_back(stringTmp);
BeginPos=Pos+2;
}
va_end(vl);
for (int i = 0; i <vtring.size(); i++)
{
std::cout<<vtring[i].c_str();
}
}
这样这个函数就可以实现简单的PrintChars("This is my %d Time to Print Msg", 100);
虽然有点土,但还是实现了简单的功能哒。
前面的这一系列操作,只是因为还没有找到一个非常强大的API,vsnprintf
有了这个,就不用前面那么费劲的去解析%啦,
参考其函数原型:
int _vsnprintf(char *buffer, size_t max_count, const char *format, va_list vArgList);
简化刚才的代码如下:
std::string PrintChars(char* szFormat, ...) {
char szMsg[8192] = {0};
va_list argp; // 这个还是一样,
va_start(argp,szFormat); // 这个也一样
int iLen = vsnprintf(szMsg, sizeof(szMsg)-1, szFormat, argp); // 这个就不一样了,非常简单
va_end(argp);
return szMsg;
}
这样就可以使用任何类型的格式化了,包括%s,%lld之类的,很好用吧,还有更好用的,我们在代码里面很多时候会用宏定义去替换一些东西,可变参数也不例外,也可用宏噢。
#define macrodebug(format, ...) PrintChars (format, ## __VA_ARGS__)