strcpy与strncpy的区别
- strcpy:字串复制
原型:char *strcpy(char *dest, char *src);
功能:把src所指由'\0'结束的字符串复制到dest所指的数组中。
说明:src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳src的字符串。
返回指向dest的指针。
注意:当src串长度>dest串长度时,程序仍会将整个src串复制到dest区域,可是dest数组已发生溢出。
因此会导致dest栈空间溢出以致产生崩溃异常。如果不考虑src串的完整性,可以把dest数组最后一元素置为NULL,从dest串长度处插入NULL截取字串。
原型:char * strncpy(char *dest, char *src, size_t n);
功能:将字符串src中最多n个字符复制到字符数组dest中(它并不像strcpy一样遇到NULL才停止复制,而是等凑够n个字符才开始复制),返回指向dest的指针。
说明:
如果n > dest串长度,dest栈空间溢出产生崩溃异常。
否则:
1)src串长度<=dest串长度,(这里的串长度包含串尾NULL字符)
如果n=(0, src串长度),src的前n个字符复制到dest中。但是由于没有NULL字符,所以直接访问dest串会发生栈溢出的异常情况。
如果n = src串长度,与strcpy一致。
如果n = dest串长度,[0,src串长度]处存放于desk字串,(src串长度, dest串长度]处存放NULL。
2)src串长度>dest串长度
如果n =dest串长度,则dest串没有NULL字符,会导致输出会有乱码。如果不考虑src串复制完整性,可以将dest最后一字符置为NULL。
综上,一般情况下,使用strncpy时,建议将n置为dest串长度(除非你将多个src串都复制到dest数组,并且从dest尾部反向操作),复制完毕后,为保险起见,将dest串最后一字符置NULL,避免发生在第2)种情况下的输出乱码问题。当然喽,无论是strcpy还是strncpy,保证src串长度<dest串长度才是最重要的。
- 附:
Strcpy和Strncpy的区别- -
第一种情况:
char* p="how are you ?";
char name[20]="ABCDEFGHIJKLMNOPQRS";
strcpy(name,p); //name改变为"how are you ? "====>正确!
strncpy(name,p, sizeof(name));//name改变为"how are you ?" =====>正确!后续的字符将置为NULL
第二种情况:
char* p="how are you ?";
char name[10];
strcpy(name,p); //目标串长度小于源串,错误!
name[sizeof(name)-1]='\0'; //和上一步组合,弥补结果,但是这种做法并不可取,因为上一步出错处理方式并不确定
strncpy(name,p,sizeof(name)); //源串长度大于指定拷贝的长度sizeof(name),注意在这种情况下不会自动在目标串后面加'\0'
name[sizeof(name)-1]='\0'; //和上一步组合,弥补结果
-
strncpy的一种实现方式
1 char * strncpy(char* dest, const char* src, int n){ 2 char c; 3 char *s = dest; 4 if (n >= 4) 5 { 6 size_t n4 = n >> 2; 7 for (;;) 8 { 9 c = *src++; 10 *dest++ = c; 11 if (c == '\0') 12 break; 13 c = *src++; 14 *dest++ = c; 15 if (c == '\0') 16 break; 17 c = *src++; 18 *dest++ = c; 19 if (c == '\0') 20 break; 21 c = *src++; 22 *dest++ = c; 23 if (c == '\0') 24 break; 25 if (--n4 == 0) 26 goto last_chars; 27 } 28 n -= dest - s; 29 goto zero_fill; 30 } 31 last_chars: 32 n &= 3; 33 if (n == 0) 34 return s; 35 for (;;) 36 { 37 c = *src++; 38 --n; 39 *dest++ = c; 40 if (c == '\0') 41 break; 42 if (n == 0) 43 return s; 44 } 45 zero_fill: 46 while (n-- > 0) 47 dest[n] = '\0'; 48 return s; 49 }