在调cgi的时候原本想写一个简单的log,但是入参始终不对,昨晚很丧气,今早起来再查看,发现了bug
之前是这么写的,arg得到的值却总是不对,然后才发现有vsprintf函数专门用来函数变参的:
int Mcgi_log(char *format, ... )
{
va_list arg;
char ac_log_time[TIME_LEN];
char fmt[1024] ="";
char cmd[1024] ="";
memset(ac_log_time, 0, TIME_LEN);
get_log_time(ac_log_time);
va_start(arg, format);
sprintf(fmt,format,arg);//这句应该改为:vsprintf(fmt,format,arg);
sprintf(cmd, "echo \"[%s]: %s\" >> /tmp/byy/cgi.log ", ac_log_time, fmt);
system(cmd);
va_end(arg);
return 0;
}
下面是摘抄:
vsprintf 函数定义如下:
int vsprintf(char *string, char *format, va_list param);
vsprintf 是sprintf 的一个变形,它只有三个参数。vsprintf 用于执行有不定数量参数的函数,类似printf 格式。vsprintf的前两个参数与sprintf相同:一个用于保存结果的字符串缓冲区和一个格式化字符串。第三个参数是指向格式化参数队列的指针。实际上,该指针指向在堆栈中供函数调用的变量。va_list、va_start和va_end宏(在STDARG.H中定义)帮助我们处理堆栈指针。本章最后的SCRNSIZE程式展示了使用这些宏的方法。使用vsprintf 函数,sprintf
函数可以这样编写:
int sprintf (char * szBuffer, const char * szFormat, ...)
{
int iReturn ;
va_list pArgs ;
va_start (pArgs, szFormat) ;
iReturn = vsprintf (szBuffer, szFormat, pArgs) ;
va_end (pArgs) ;
return iReturn ;
}
va_start宏将pArg设置为指向一个堆栈变量,该变量位址在堆栈参数szFormat的上面。
由於许多Windows早期程序使用了sprintf 和vsprintf,最终导致Microsoft 向Windows API中增添了两个相似的函数。Windows的wsprintf 和wvsprintf 函数在功能上与sprintf 和vsprintf 相同,但它们不能处理浮点格式。
当然,随著宽字符的发表,sprintf 类型的函数增加了许多,使得函数名称变得极为混乱。下面列出了Microsoft 的C执行时期函数库和Windows支持的所有sprintf 函数。
sprintf(fmt,format,arg);//这句应该改为:vsprintf(fmt,format,arg);