strcpy和memcpy都是标准C库函数,它们有下面的特点。
strcpy提供了字符串的复制。即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制字符串的结束符。
已知strcpy函数的原型是:char* strcpy(char* dest, const char* src);
memcpy提供了一般内存的复制。即memcpy对于需要复制的内容没有限制,因此用途更广。
void *memcpy( void *dest, const void *src, size_t count );
?
char * strcpy ( char * dest, const char * src) { if ((src == NULL) || (dest == NULL)) { return NULL; } char *strdest = dest; while ((*strDest++ = *strSrc++)!= '\0' ); return strdest; } void * memcpy ( void *memTo, const void *memFrom, size_t size) { if ((memTo == NULL) || (memFrom == NULL)) return NULL; char *tempFrom = ( char *)memFrom; char *tempTo = ( char *)memTo; while (size -- > 0) *tempTo++ = *tempFrom++ ; return memTo; } |
strcpy和memcpy主要有以下3方面的区别。
1、复制的内容不同。strcpy只能复制字符串,而memcpy可以复制任意内容,例如字符数组、整型、结构体、类等。
2、复制的方法不同。strcpy不需要指定长度,它遇到被复制字符的串结束符"\0"才结束,所以容易溢出。memcpy则是根据其第3个参数决定复制的长度。
3、用途不同。通常在复制字符串时用strcpy,而需要复制其他类型数据时则一般用memcpy
------------------------------------------
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 ? " ====>正确!
第二种情况:
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'; //和上一步组合,弥补结果
总结:strcpy
源字串全部拷贝到目标字串中,包括'\0',但是程序员必须保证目标串长度足够,且不与源串重叠。
strncpy
如果目标长>=指定长>源长,则将源串全部拷贝到目标串,连同'\0'
如果指定长<源长,则将截取源串中按指定长度拷贝到目标字符串,不包括'\0'
如果指定长>目标长,错误!
char * strncpy ( char *dest, char *src,size_tn); |
功能:
(c/c++)复制字符串src中的内容(字符,数字、汉字....)到字符串dest中,复制多少由size_t的值决定,返回指向dest的指针。如果遇到空字符('\0')[1] ,则空字符后面全部为空(字符),下面举例说明[2] : ?
1 2 3 4 5 6 7 | dest[]= "Hell99iam!" ; src[]= "abc\0def" ; strncpy (dest,src,5); |
此时,dest区域是这样的:'a','b','c','\0','\0','9','i','a','m','!' 注意:\0,\0并不是添加在!的后面。
头文件:
?
说明:
如果n > dest串长度,dest栈空间溢出产生崩溃异常。 否则: 1)src串长度<=dest串长度,(这里的串长度包含串尾NULL字符) 如果n<src串长度,dest[..]=[src1, srcN]。由于长度达到srcNULL,所以直接访问dest串会发生栈溢出的异常情况,有些变态,一般情况下都不会这么做(即:lenC, lenB, lenA: NG)。 如果n = src串长度,与strcpy一致, (即:lenC, lenB, lenB: OK)。 如果n >src串长度,dest[...]=[0, srcNULL] +[NULL...],OK。(即:lenC, lenB, lenC: OK) 2)src串长度>dest串长度 如果n =dest串长度,则dest串没有NULL字符,会导致输出会有乱码。如果不考虑src串复制完整性,可以将dest 最后一字符置为NULL。(即:lenB, lenC, lenB: NG) 综上,一般情况下,使用strncpy时,建议将n置为dest串长度(除非你将多个src串都复制到dest数组,并且从dest尾部反向操作),复制完毕后,为保险起见,将dest串最后一字符置NULL,避免发生在第2)种情况下的输出乱码问题。当然喽,无论是strcpy还是strncpy,保证dest串容量(能容纳下src串)才是最重要的。
?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 | char * strncpy ( char *dest, const char *src, int n) { char c; char *s = dest; if (n >= 4) { size_t n4 = n >> 2; for (;;) { c = *src++; *dest++ = c; if (c == '\0' ) break ; c = *src++; *dest++ = c; if (c == '\0' ) break ; c = *src++; *dest++ = c; if (c == '\0' ) break ; c = *src++; *dest++ = c; if (c == '\0' ) break ; if (--n4 == 0) goto last_chars; } n -= dest - s; goto zero_fill; } last_chars: n &= 3; if (n == 0) return s; for (;;) { c = *src++; --n; *dest++ = c; if (c == '\0' ) break ; if (n == 0) return s; } zero_fill: while (n-- > 0) dest[n] = '\0' ; return s; } |
3与strcpy的区别编辑
附: Strcpy和Strncpy的区别- - 第一种情况: ?
1 2 3 4 | char * p= "how are you ?" ; char name[20]= "ABCDEFGHIJKLMNOPQRS" ; strcpy (name,p); strncpy (name,p, sizeof (name)); |
第二种情况: ?
1 2 3 4 5 6 | char * p= "how are you ?" ; char name[10]; strcpy (name,p); name[ sizeof (name)-1]= '\0' ; strncpy (name,p, sizeof (name)); name[ sizeof (name)-1]= '\0' ; |
What Doesn't Kill Me Makes Me Stronger