snprintf

http://baike.baidu.com/view/1966670.htm

http://blog.csdn.net/yefengnidie/article/details/6248319

snprintf

  int snprintf(char *str, size_t size, const char *format, ...);
  将可变个参数(...)按照format格式化成字符串,然后将其复制到str中
  (1) 如果格式化后的字符串长度 < size,则将此字符串全部复制到str中,并给其后添加一个字符串结束符('\0');
  (2) 如果格式化后的字符串长度 >= size,则只将其中的(size-1)个字符复制到str中,并给其后添加一个字符串结束符('\0')
  函数返回值:若成功则返回欲写入的字符串长度,若出错则返回负值。 --------------------------------------------
  #include <stdio.h> //snprintf()
  #include <string.h> //strlen()
  int main()
  {
  char toname[30];
  char *name = "yangtaiping";
  printf("strlen(name) = %d\n", strlen(name));
  snprintf(toname, strlen(name)+strlen("姓名:")+1, "姓名:%s", name);
  printf("toname = %s\n", toname);
  printf("strlen(name) = %d\n", strlen(name));
  }
  strcpy() sprintf() strcat() 存在安全隐患, 其对应的安全版为:strncpy() snprintf() strncat() 。 
  snprintf(s, 100, "%.*S", 3, "abcd");s的值为abc %.*s 表示有两项, 第一项指定了长度,第二项则是%s的内容,所以取前三位 词条图册更多图册
  类似的函数还有:
  #include <stdio.h>
  int printf(const char *format, ...); //格式化输出到标准输出设备
  int fprintf(FILE *stream,const char *format, ...); //格式化输出到文件流
  int sprintf(char *str,const char *format, ...); //格式化的输出字符串输出到str指向的地址空间
////////////////////////////////////////////////////////////////////////////

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将没有'/0'结尾

2.
char sa[256];
_snprintf(sa,sizeof(sa)-1,"%s",sb);
//错误原因:当sb的长度>=255的时候,sa将没有'/0'结尾,忘记给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;
}

Linux and Windows snprintf区别 收藏 
今天使用snprintf函数时想到了: 
    strcpy, strncpy
    strcmp, strncmp
    strcat, strncat
    sprintf, snprintf

最喜欢用的就是snprintf, 因为它会自动在后面加'/0'. 在网上看了一下. 原来VC中_snprintf不是这样的.

VC中的_snprintf函数并没有按照这样的规定来做,它在输出缓冲区不够大时就不会输出结尾的'/0'(跟strncpy的行为类似)。所以要让上面的程序工作正常,必须做相应的修改。

  char buf[5];
  _snprintf(buf, 5, "This is a test string."); // buf becomes "This ", buf[4] is ' '
  buf[4] = 0;                              // buf[4] is '/0' now. 为了安全, 在windows下要手动添加
  
  _snprintf(buf, 6, "This is a test string."); // ERROR: buffer overflow
  _snprintf(buf, 5, "abc"); // buf becomes "abc", the value of buf[3] is '/0', buf[4] is undefined.

如果要保证可移植性,就得按VC的写法,做一次对于标准来说很多余的"填0"的操作,再通过定义宏来区别平台分别选用snprintf或_snprintf。

一个小的测试代码:
可以看到有两个地方不一样:
    1>. 结尾的'/0'处理!
    2>. 函数返回值不一样!
#include <stdio.h>
#include <string.h>

#define STR_LEN 32

#ifdef _WIN32
#define snprintf _snprintf
#endif

/**
 * dump string
 */
static void dump_str(const char *title, const char *str, int len)
{
  int i;

  printf("%s[%d].%p /n", title, len, str);
  for (i = 0; i < len; i++)
    printf("<%c,%02x>", str[i], (unsigned char)str[i]);

  printf("/n");
}

int main(int argc, char **argv)
{
  int t;
  char str1[STR_LEN], str2[STR_LEN];

  strcpy(str1, "123456789");
  dump_str("str1", str1, STR_LEN);

  t = snprintf(str2, STR_LEN, "%s", str1);
  dump_str("str2", str2, STR_LEN);
  printf("snprintf() ret.%d /n", t);

  t = snprintf(str2, 5, "%s", str1);
  dump_str("str2", str2, STR_LEN);
  printf("snprintf() ret.%d /n", t);

  return (0);
}

/*
 Test Env:
   gcc version 3.4.5 20051201 (Red Hat 3.4.5-2)

  Result:
str1[32].0xbffb9190
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><?fb><?bf><?d5><?82><,04><,08>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00><?c8><?91><?fb><?bf><&,26><?85><,04><,08>
str2[32].0xbffb9170
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><w,77><,01>< ,00><?92><?fb><?bf><?f8><d,64><?ad>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>
snprintf() ret.9
str2[32].0xbffb9170
<1,31><2,32><3,33><4,34>< ,00><6,36><7,37><8,38><9,39>< ,00><w,77><,01>< ,00><?92><?fb><?bf><?f8><d,64><?ad>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>
snprintf() ret.9

 -------------------------------------------------------------------------
 Test Env:
   Microsoft Windows XP [版本 5.1.2600]
   Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 12.00.8168 for 80x86


 Result:
str1[32].0012FF60
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><,12>< ,00><,10><,1d><@,40>< ,00>< ,00><,08>< ,00>< ,00><,04>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00><?e0><,1e><@,40>< ,00>
str2[32].0012FF3C
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><?93><|,7c><&#63733;,ff><&#63733;,ff><&#63733;,ff><&#63733;,ff><?eb><,06><?93><|,7c><?ef><0,30><@,40>< ,00>< ,00>< ,00><7,37>< ,00><    ,09>< ,00>< ,00>< ,00>
snprintf() ret.9
str2[32].0012FF3C
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><?93><|,7c><&#63733;,ff><&#63733;,ff><&#63733;,ff><&#63733;,ff><?eb><,06><?93><|,7c><?ef><0,30><@,40>< ,00>< ,00>< ,00><7,37>< ,00><    ,09>< ,00>< ,00>< ,00>
snprintf() ret.-1
 */



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值