Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4042801
  • 博文数量: 536
  • 博客积分: 10470
  • 博客等级: 上将
  • 技术积分: 4825
  • 用 户 组: 普通用户
  • 注册时间: 2006-05-26 14:08
文章分类

全部博文(536)

文章存档

2024年(3)

2021年(1)

2019年(1)

2017年(1)

2016年(2)

2013年(2)

2012年(10)

2011年(43)

2010年(10)

2009年(17)

2008年(121)

2007年(252)

2006年(73)

分类:

2007-08-30 18:07:09

今天使用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><,04><,08>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00>< ,00><&,26><,04><,08>
str2[32].0xbffb9170
<1,31><2,32><3,33><4,34><5,35><6,36><7,37><8,38><9,39>< ,00><,01>< ,00>< ,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><,01>< ,00>< ,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><,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><|,7c><,ff><,ff><,ff><,ff><,06><|,7c><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><|,7c><,ff><,ff><,ff><,ff><,06><|,7c><0,30><@,40>< ,00>< ,00>< ,00><7,37>< ,00><    ,09>< ,00>< ,00>< ,00>
snprintf() ret.-1
 */

阅读(5297) | 评论(0) | 转发(0) |
0

上一篇:CPU--基本概念

下一篇:Linux find command

给主人留下些什么吧!~~