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) ​​​​// 实现src到dest的复制​​​​{​​​​​​​​if​​​​((src == NULL) || (dest == NULL)) ​​​​//判断参数src和dest的有效性​​​​ {​​​​​​​​return​​​​NULL;​​​​ }​​​​​​​​char​​​​*strdest = dest; ​​​​//保存目标字符串的首地址​​​​​​​​while​​​​((*strDest++ = *strSrc++)!=​​​​'\0'​​​​); ​​​​//把src字符串的内容复制到dest下​​​​​​​​return​​​​strdest;​​​​}​​​​void​​​​*​​​​memcpy​​​​(​​​​void​​​​*memTo, ​​​​const​​​​void​​​​*memFrom, ​​​​size_t​​​​size)​​​​{​​​​​​​​if​​​​((memTo == NULL) || (memFrom == NULL)) ​​​​//memTo和memFrom必须有效​​​​​​​​return​​​​NULL;​​​​​​​​char​​​​*tempFrom = (​​​​char​​​​*)memFrom; ​​​​//保存memFrom首地址​​​​​​​​char​​​​*tempTo = (​​​​char​​​​*)memTo; ​​​​//保存memTo首地址 ​​​​​​​​while​​​​(size -- > 0) ​​​​//循环size次,复制memFrom的值到memTo中​​​​​​​​ *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并不是添加在!的后面。

头文件:

​?​

1

​#include<string.h>​


说明:

如果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串)才是最重要的。

2实现方式​​编辑​

​?​

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); ​​​​//name改变为"how are you ? "====>正确! ​​​​strncpy​​​​(name,p, ​​​​sizeof​​​​(name));​​​​//name改变为"how are you ?" =====>正确!后续的字符将置为NULL​

第二种情况: ​​?​

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)); ​​​​//源串长度大于指定拷贝的长度sizeof(name),注意在这种情况下不会自动在目标串后面加'\0' ​​​​name[​​​​sizeof​​​​(name)-1]=​​​​'\0'​​​​; ​​​​//和上一步组合,弥补结果​


What Doesn't Kill Me Makes Me Stronger