zoukankan      html  css  js  c++  java
  • C语言snprintf函数

    int snprintf(char *restrict buf, size_t n, const char * restrict  format, ...);

    函数说明:最多从源串中拷贝n1个字符到目标串中,然后再在后面加一个0。所以如果目标串的大小为n

                     的话,将不会溢出。

    函数返回值:若成功则返回欲写入的字符串长度,若出错则返回负值。

    Result1(推荐的用法)

    #include <stdio.h>
    #include <stdlib.h>

    int main()
    {
         char str[10]={0,};
         snprintf(str, sizeof(str), "0123456789012345678");
         printf("str=%s ", str);
         return 0;
    }

     

    root] /root/lindatest
    $ ./test
    str=012345678

     

     

    Result2:(不推荐使用)

    #include <stdio.h>
    #include <stdlib.h>

    int main()
    {
        char str[10]={0, };
        snprintf(str, 18, "0123456789012345678");
        printf("str=%s ", str);
        return 0;
    }

     

    root] /root/lindatest
    $ ./test
    str=01234567890123456

     

     

    snprintf函数返回值的测试:

    #include <stdio.h>
    #include <stdlib.h>

    int main()
    {
        char str1[10] ={0, };
        char str2[10] ={0, };
        int ret1=0,ret2=0;
        ret1=snprintf(str1, sizeof(str1), "%s", "abc");
        ret2=snprintf(str2, 4, "%s", "aaabbbccc");
        printf("aaabbbccc length=%d ", strlen("aaabbbccc"));
        printf("str1=%s,ret1=%d ", str1, ret1);
        printf("str2=%s,ret2=%d ", str2, ret2);
        return 0;
    }

    [root] /root/lindatest
    $ ./test
    aaabbbccc length=9
    str1=abc,ret1=3
    str2=aaa,ret2=9

    ***************************************************************************

    关于sprintf和snprintf的正确使用。

    考虑以下有缺陷的例子:
    void f(const char *p)
    {
        char buf[11]={0};
        sprintf(buf,"%10s",p); // very dangerous
        printf("%sn",buf);
    }

    不要让格式标记“%10s”误导你。如果p的长度大于10个字符,那么sprintf() 的写操作就会越过buf的边界,从而产生一个缓冲区溢出。
    检测这类缺陷并不容易,因为它们只在 p 的长度大于10个字符的时候才会发生。黑客通常利用这类脆弱的代码来入侵看上去安全的系统。

    要修正这一缺陷,可以使用函数snprintf()代替函数sprintf()。

    函数原型:int snprintf(char *dest, size_t n, const char *fmt, ...);
    函数说明: 最多从源串中拷贝n-1个字符到目标串中,然后再在后面加一个0。所以如果目标串的大小为n的话,将不会溢出。
    函数返回值: 若成功则返回存入数组的字符数,若编码出错则返回负值。

    推荐的用法:
    void f(const char *p)
    {
        char buf[11]={0};
        snprintf(buf, sizeof(buf), "%10s", p); // 注意:这里第2个参数应当用sizeof(str),而不要使用硬编码11,也不应当使用sizeof(str)-1或10
        printf("%sn",buf);

    **************************************************************************

    众所周知,sprintf不能检查目标字符串的长度,可能造成众多安全问题,所以都会推荐使用snprintf.

    snprintf(_snprintf)的声明是这样的

    int _snprintf(
       char *buffer,
       size_t count,
       const char *format [,
          argument] ...
    );

    If len < count, then len characters are stored in buffer, a null-terminator is appended, and len is returned.

    If len = count, then len characters are stored in buffer, no null-terminator is appended, and len is returned.

    If len > count, then count characters are stored in buffer, no null-terminator is appended, and a negative value is returned.


    最常见的错误用法有:
    1.
    char sa[256]={0};
    _snprintf(sa,sizeof(sa),"%s",sb);
    //错误原因:当sb的长度>=256的时候,sa将没有''结尾

    2.
    char sa[256];
    _snprintf(sa,sizeof(sa)-1,"%s",sb);
    //错误原因:当sb的长度>=255的时候,sa将没有''结尾,忘记给sa初始化

    3.
    char sa[256];
    _snprintf(sa,sizeof(sa)-1,"%s",sb);
    sa[sizeof(sa)]=0;
    //错误原因:最后一行数组越界

    正确的用法
    1. //推荐用法
    char sa[256];
    sa[sizeof(sa)-1]=0;
    _snprintf(sa,sizeof(sa),"%s",sb);
    if(sa[sizeof(sa)-1]!=0)
    {
       printf("warning:string will be truncated");
       sa[sizeof(sa)-1]=0;
    }

    2.
    char sa[256]={0};
    int result = _snprintf(sa,sizeof(sa),"%s",sb);
    if(result==sizeof(sa) || result<0)
    {
        printf("warning:sting will be truncated");
       sa[sizeof(sa)-1]=0;
    }

  • 相关阅读:
    微软ASP.NET网站部署指南(4):配置项目属性
    iOS 设计模式之抽象工厂
    How can I move a MySQL database from one server to another?
    CentOS 7 上安装vim(默认未安装)
    How to resize slide dimensions without resizing any objects on the slide?
    CentOS 7.3 上安装docker
    美国留学访学(访问学者)必备信用卡
    西安理工大学税务登记证、银行账号信息
    Oracle联合多个子查询(inner join)
    linux tail
  • 原文地址:https://www.cnblogs.com/lixiaofei1987/p/3171492.html
Copyright © 2011-2022 走看看