类似printf这样不定输入参数的函数的实现
现实中,用过printf函数,觉得非常有意思,这个参数可以不定的,可以0个参数,也可以多个参数,都是用这样一个函数
很神奇
那么,他们怎么实现的?如果我要实现一个类似功能函数,又如何操作?
不定参数在一些特殊场合下是很有用的,例如在字符串的格式化合成,像printf()函数、日志输出等场合都很容易想到这样的结构。
其实,这种方法在C语言里就已经存在了,也不是什么新奇特的东西,一般要实现这样的功能,需要以下几个函数/类型组成:
va_list 、va_start、va_avg、va_end
头文件:stdarg.h
先来看下,他们是怎么实现的:
//获取N个数之和
uint GetSum(uint nD1,...)
{
va_list st;
va_start(st, nD1);
uint total = nD1;
uint tmp;
while((tmp = va_arg(st, uint)) != 0)
{
total += tmp;
}
va_end(st);
return total;
}
//例2,也是+++++,但处理方式不一样了
double sum_series(int num, ...)
{
double sum = 0.0, t;
va_list argptr;
va_start(argptr, num);//初始化变元指针
while (num--)
{
t = va_arg(argptr, double);
sum = sum + t;
}
va_end(argptr);
return sum;
}
先来说说,这两种方式的缺点吧:
例1,需要输入一个“0”作为结束符,而且中间参数中不能出现0;这个条件限制,另外输入的参数必须是同一种类型,不是非常灵活。
例2,需要预先输入参数的个数,这里也有严重缺陷的,因为如果输出的num大于实际后面的参数个数的话,就越界了;另外参数增加就要修改num的输入值,这样也不方便;输入的参数必须是同一种类型,不是非常灵活。
问题,printf可是没有要求输入参数的个数以及类型必须统一,这样的要求哦?
的确,printf很灵活,没有那么限制,那么如何实现呢?难道va_arg存在特殊判断可以确定最后一个参数?
遗憾的是,va_arg是不知道最后一个参数的,除非像例子里一样,告之个数或特殊参数为结尾。。。
那么printf怎么做到的?
猜测,printf的第一个参数为 char* ,是个字符串,而且输入的时候,有参数的时候必须以“%”为开头,
那么,问题就简单了,我们先遍历一遍,看看%出现几次(需排除%%转义),不就是知道了个数了。有了个数,后面的都好处理。printf,我们在第一个参数里,输入%个数超过后面的参数,此时会出错的!但如果%的个数小于后面的参数,却没有问题。
然后在解析下%后面的s d什么的,就不是问题了。。。。
最后,来一段自己写的例子:
void DEBUG_LOG(uint nLogID,char *strFmt,...)
{
char strLog[2048];
//查找“%”
va_list lsID;
va_start(lsID, strFmt);
vsprintf(strLog, strFmt, lsID);
va_end(lsID);
DEBUG_LOG00(nLogID,strLog);
}