依据源串的\0作为结束判断的,不会检查需要拷贝的缓冲区的大小,如果目标空间不够,就有溢出问题。
原型
char * strncpy(char *dest, char *src, size_t n);
复制字符串src中的内容(字符,数字、汉字....)到字符串dest中,复制多少由size_t的值决定,返回指向dest的指针。如果遇到空字符('\0'),则空字符后面为原来的字符
代码如下:
dest[]="Hell99iam!";
src[]="abc\0def";
strncpy(dest,src,5);
此时,dest区域是这样的:'a','b','c','\0','\0','9','i','a','m','!'
而非这样:'a','b','c','\0','\0','\0','\0','\0','\0','\0'
'\0','\0'并不是添加在'!'的后面。可以用for循环输出每个元素的ascii来确定。
strncpy是字符串拷贝推荐的用法
原型:
errno_t strncpy_s(
char *strDest,
size_t numberOfElements,
const char *strSource,
size_t count
);
参数numberOfElements表明dest中的字节数,防止目标指针dest中的空间不够,同时返回值改成返回错误代码,而不是返回char*。
会在字符串结束处填补一个空字符。
count参数需要小于目标缓冲区大小。
如果情况如下代码:
char dst[5];
strncpy_s(dst, sizeof(dst), "a long string", 5);
表示使用strncpy_s 拷贝5个字节到dst缓冲区,使得没用多余空间给空字符结束符,填补空字符时会溢出,并调用异常处理句柄。
需要补空字符时使用 _TRUNCATE 或者 (size – 1)
代码如下:
char dst[5];
strncpy_s(dst, sizeof(dst), "a long string", _TRUNCATE);
strncpy_s(dst, sizeof(dst), "a long string", sizeof(dst)-1);
如果源地址字符串和目标地址字符串重叠,则结果未定义。
_s版本函数并不是标准库,不推荐使用
4、snprintf
原型:
int snprintf(char *str, size_t size, const char *format, ...)
将可变个参数(...)按照format格式化成字符串,然后将其复制到str中
(1) 如果格式化后的字符串长度 < size,则将此字符串全部复制到str中,并给其后添加一个字符串结束符('\0');
(2) 如果格式化后的字符串长度 >= size,则只将其中的(size-1)个字符复制到str中,并给其后添加一个字符串结束符('\0'),返回值为欲写入的字符串长度。
#include
int main () {
char a[16];
size_t i;
i = snprintf(a, 13, "%012d", 12345); // 第 1 种情况
printf("i = %lu, a = %s\n", i, a); // 输出:i = 12, a = 000000012345
i = snprintf(a, 9, "%012d", 12345); // 第 2 种情况
printf("i = %lu, a = %s\n", i, a); // 输出:i = 12, a = 00000001
return 0;
}