C99标准函数snprintf在不同平台的实现差异

snprintf函数原型

C99提供的库函数snprintf的原型为:

int snprintf ( char * str, size_t size, const char * format, ... );

其含义如下:

  • str:格式化后输出的目标字符串
  • size:常见有两种实现,一是允许写入字符串的长度,不含\0;二是允许写入的字节数,含\0
  • format:格式化字符串
  • …:可变参数
  • 返回值:写入字符串无截断时返回写入str的字符串长度,不含\0;写入字符串有截断时常见有两种实现,一是返回需要写入str的字符串长度,不含\0,二是返回-1。

由于snprintf在不同平台下的实现可能不同,在跨平台使用时需要注意其实现方式。笔者针对VC2015、MinGW-8.5.0及stb库中的实现方式进行了实验,具体如下文所述。

size小于目标字符串长度时

示例代码:

    char buf[12] = { 0 };
    char *str = "Hello world!";
    size_t sz = 3;
    int len = snprintf(buf, sz, "%s", str);
    printf("snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = _snprintf(buf, sz, "%s", str);
    printf("_snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = stbsp_snprintf(buf, sz, "%s", str);
    printf("stbsp_snprintf: len=%d, and buf=[%s]\n", len, buf);

使用VC2015运行上述代码的输出结果为:

snprintf: len=12, and buf=[He]
_snprintf: len=-1, and buf=[Hel]
stbsp_snprintf: len=12, and buf=[He]

使用MinGW-8.5.0运行上述代码的输出结果为:

snprintf: len=-1, and buf=[Hel]
_snprintf: len=-1, and buf=[Hel]
stbsp_snprintf: len=12, and buf=[He]

size等于目标字符串长度时

示例代码:

    char buf[12] = { 0 };
    char *str = "Hello world!";
    size_t sz = 11;
    int len = snprintf(buf, sz, "%s", str);
    printf("snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = _snprintf(buf, sz, "%s", str);
    printf("_snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = stbsp_snprintf(buf, sz, "%s", str);
    printf("stbsp_snprintf: len=%d, and buf=[%s]\n", len, buf);

使用VC2015运行上述代码的输出结果为:

snprintf: len=11, and buf=[Hello worl]
_snprintf: len=11, and buf=[Hello world]
stbsp_snprintf: len=11, and buf=[Hello worl]

size等于目标字符串长度+1时

示例代码:

    char buf[12] = { 0 };
    char *str = "Hello world!";
    size_t sz = 12;
    int len = snprintf(buf, sz, "%s", str);
    printf("snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = _snprintf(buf, sz, "%s", str);
    printf("_snprintf: len=%d, and buf=[%s]\n", len, buf);

    len = stbsp_snprintf(buf, sz, "%s", str);
    printf("stbsp_snprintf: len=%d, and buf=[%s]\n", len, buf);

使用VC2015运行上述代码的输出结果为:

snprintf: len=11, and buf=[Hello world]
_snprintf: len=11, and buf=[Hello world]
stbsp_snprintf: len=11, and buf=[Hello world]

使用MinGW-8.5.0运行上述代码的输出结果为:

snprintf: len=11, and buf=[Hello world]
_snprintf: len=11, and buf=[Hello world]
stbsp_snprintf: len=11, and buf=[Hello world]

实验结果

经过上述实验可知:

实现size参数返回值
snprintf(VC2015)str中允许使用的字节数,包含\0,写入的字符数最大为size-1无论是否截断,均返回需要写入的字符串长度
snprintf(MinGW8.5.0)str中允许写入的字符数,不含\0,写入的字符数最大为size无截断时返回需要写入的字符串长度,截断时返回-1
_snprintfstr中允许写入的字符数,不含\0,写入的字符数最大为size无截断时返回需要写入的字符串长度,截断时返回-1
stbsp_snprintfstr中允许使用的字节数,包含\0,写入的字符数最大为size-1无论是否截断,均返回需要写入的字符串长度

不同平台下snprintf的正确使用方式为:

实现size参数返回值
snprintf(VC2015)size最小为目标字符串长度+1,不超过str的内存长度返回值不小于size表示截断
snprintf(MinGW8.5.0)size最小为目标字符串长度,不超过str的内存长度-1返回-1表示截断
_snprintfsize最小为目标字符串长度,不超过str的内存长度-1返回-1表示截断
stbsp_snprintfsize最小为目标字符串长度+1,不超过str的内存长度返回值不小于size表示截断

若本实验中的结论与你在网上查阅的资料不一致,建议自己做实验实测一下为好。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值