一杯清酒邀明月
天下本无事,庸人扰之而烦耳。

一、定义

众所周知,sprintf不能检查目标字符串的长度,可能造成众多安全问题,所以都会推荐使用snprintf。自从snprintf代替了sprintf,相信大家对snprintf的使用都不会少,函数定义如下:

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

C 库函数 int snprintf(char *str, size_t size, const char *format, ...) 设将可变参数(...),按照 format 格式化成字符串,并将字符串复制到 str 中,size 是要写入的字符的最大数目,超过 size 会被截断。

参数
str -- 目标字符串。
size -- 拷贝字节数(Bytes)。
format -- 格式化成字符串。
... -- 可变参数。
返回值
(1) 如果格式化后的字符串长度小于等于 size,则会把字符串全部复制到 str 中,并给其后添加一个字符串结束符 \0,
(2) 如果格式化后的字符串长度大于 size,超过 size 的部分会被截断,只将其中的 (size-1) 个字符复制到 str 中,并给其后添加一个字符串结束符 \0,
(3)返回值为欲写入的字符串长度,若出错则返回负值。
二、实例

 1  int main()
 2  {
 3      char a[16];
 4      size_t i;
 5  
 6      i = snprintf(a, 13, "%012d", 12345);  // 第 1 种情况,12345复制到数组a中
 7      printf("i = %lu, a = %s\n", i, a);    // 输出:i = 12, a = 000000012345, 12位
 8  
 9      i = snprintf(a, 9, "%012d", 12345);   // 第 2 种情况,超出的截断,只复制1到数组a中
10      printf("i = %lu, a = %s\n", i, a);    // 输出:i = 12, a = 00000001,9位
11  
12          return 0;
13 }

 其他例子

 1  std::string printTime(const timeval &tv) {
 2      time_t sec_tmp = tv.tv_sec;
 3      struct tm *tm = localtime(&sec_tmp);
 4      char buf[128];
 5      snprintf(buf, sizeof(buf), "%d-%02d-%02d %02d:%02d:%02d.%03d",
 6          1900 + tm->tm_year,
 7          1 + tm->tm_mon,
 8          tm->tm_mday,
 9          tm->tm_hour,
10          tm->tm_min,
11          tm->tm_sec,
12          (int)(tv.tv_usec / 1000));
13      return buf;
14  }

另一个例子

假设我想将一个char类型的变量打印成2位的10进制,我应该怎么写呢?

1      size_t i;
2      char a = 'a';
3      char str[20];
4      i=snprintf(str,sizeof(str), "%02d", a);
5      printf("i = %lu, a = %s\n", i, atr);

运行结果:

解释如下:

size是限定最终生成的dest的字符数,最多拷贝size-1个字符; 一般情况下size会取sizeof(dest),这是为了dest不溢出.

在snprintf(dest, size, "str: %s\n", src)中如果size-1大于等于"str: %s\n"的长度,则把"str: %s\n"都拷贝到dest; 如果size-1小于"str: %s\n"的长度,则从"str: %s\n"拷贝size-1长度的字符串到dst,并且末尾置\0.

就是说,拷贝的长度是size-1和源字符串长度的最小值;

posted on 2022-07-19 15:21  一杯清酒邀明月  阅读(854)  评论(0编辑  收藏  举报