printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别

最近在为新做的项目找个好用的第三方日志库,感觉log4cpp挺好用的,日志输出使用了格式化字符串,不同的函数功能不同,而且变异版本比较多,自己都有点搞昏了,网上搜索也比较笼统,下面是自己coding测试格式化字符串的过程(vs2015,win10  x64环境),供参考

有C语言写作历史的程序员往往特别喜欢printf 函数。即使可以使用更简单的命令(例如puts),但printf 出现在Kernighan和Ritchie的「hello, world」程序中一点也不会令人惊奇。我们知道,增强后的「hello, world」最终还是需要printf 的格式化输出,因此我们最好从头开始就使用它。

printf 函数说明如下:

int printf (const char * szFormat, ...);
第一个参数是一个格式化字符串,后面是与格式化字符串中的代码相对应的不同类型多个参数。

这个函数比较简单,如下,没什么好说的

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第1张图片  

sprintf 函数定义如下:

int sprintf (char * szBuffer, const char * szFormat, ...);
第一个参数是字符串缓冲区,后面是一个格式字串。sprintf不是将格式化结果标准输出,而是将其存入szBuffer。该函数返回该字符串的长度。

sprintf函数vs2015提示该函数可能不安全,建议使用sprintf_s函数

我们使用sprintf_s如下:

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第2张图片

几乎每个人都经历过,当格式化字符串与被格式化的变数不合时,可能使printf 执行错误并可能造成程序当掉。使用sprintf 时,您不但要担心这些,而且还有一个新的负担:您定义的字符串缓冲区必须足够大以存放结果。Microsoft 专用函数_snprintf 解决了这一问题,此函数引进了另一个参数,表示以字节计算的缓冲区大小。

同样_snprintf函数vs2015也提示该函数可能不安全,建议使用_snprintf_s函数

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第3张图片

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第4张图片

可以看到输出字符串被截断了。

重点来了

vsprintf 函数定义如下:

int vsprintf(char *string, char *format, va_list param);

vsprintf 是sprintf 的一个变形,它只有三个参数。vsprintf 用于执行有不定数量参数的函数,类似printf 格式。vsprintf的前两个参数与sprintf相同:一个用于保存结果的字符串缓冲区和一个格式化字符串。第三个参数是指向格式化参数队列的指针。实际上,该指针指向在堆栈中供函数调用的变量。va_list、va_start和va_end宏(在STDARG.H中定义)帮助我们处理堆栈指针。va_start宏将pArg设置为指向一个堆栈变量,该变量位址在堆栈参数szFormat的上面。

同样vsprintf 函数vs2015也提示该函数可能不安全,我们使用vsprintf_s函数

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第5张图片

在寻找第三方日志库的过程中,格式化字符串需要计算当前格式化字符串的长度,如果长度不够可能越界发生崩溃,接下来这个函数能帮我们获取格式化字符串长度;

_vscprintf函数定义如下:

int _vscprintf(const char *format,va_list argptr);
Returns the number of characters in the string referenced by a list of arguments
大概意思就是说:返回格式化列表字符字节数

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第6张图片

这样就可以按照格式化字符串的大小动态申请缓冲区的大小,方便又实用,不会因为固定缓冲区大小导致崩溃的 问题发生

这个函数解决了本次格式化字符串的问题~~

还有两个函数

vsnprintf,_vsnprintf_s,参考如下验证:

printf,sprintf,sprintf_s,_snprintf_s,vsprintf,vsnprintf,_vsnprintf_s,_vscprintf等格式化字符串使用与区别_第7张图片

由于许多Windows早期程序使用了sprintf 和vsprintf,最终导致Microsoft 向Windows API中增添了两个相似的函数。

当然,随著宽字符的发表,sprintf 类型的函数增加了许多,使得函数名称变得极为混乱。类似的函数还有很多变种,有兴趣的朋友自行度娘

https://download.csdn.net/download/xianzongtanxun/10605842​​​​​​​ 放上例子代码供参考

你可能感兴趣的:(网络)