strcpy与memcpy以及strncpy

1. strcpy函数

strcpy提供了字符串的复制。即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制字符串的结束符。

函数原型及实现:

[c-sharp] view plain copy print ?
  1. char* strcpy(char* strDst, const char* strSrc)
  2. {
  3. assert((strDst != NULL) && (strSrc != NULL));
  4. char* address = strDst;
  5. while((*strDst++ = *strSrc++) != '/0');
  6. return address;
  7. }

注意:

1. 该函数会连同字符串的结束字符'/0'一起拷贝到字符数组中,故字符数组的长度应该至少是字符串的长度加1。

例:

char str[] = "123456";

这里str是一个字符数组,它存放了一个字符串"123456",由于字符串还有一个结束符"/0",所以此数组的长度为7而不是6。

char *str = "123456"; //该字符串的长度为6(字符串以字符'/0'结尾,不计入字符串长度)

char dst[7]; //字符数组的长度必须至少为字符长度加1,如6+1=7。

strcpy(dst, str);

2. 该函数会拷贝字符串的内容直至遇到第一个字符值为0的字符(第一个字符值为0的字符也会被拷贝),故如果一个字符串或字符数组含

有大量的值为0的数据,则不宜用该函数进行拷贝操作,可用memcpy函数。

2. memcpy函数

memcpy提供了一般内存的复制。即memcpy对于需要复制的内容没有限制,因此用途更广。

该函数将一个大小为size的内存段的值拷贝到另一内存段中。

其实现如下:

[c-sharp] view plain copy print ?
  1. void *memcpy(void *memTo, const void *memFrom, size_t size)
  2. {
  3. assert((memTo != NULL) && (memFrom != NULL)); //memTo和memFrom必须有效
  4. char *tempFrom = (char *)memFrom; //保存memFrom首地址
  5. char *tempTo = (char *)memTo; //保存memTo首地址
  6. while(size -- > 0) //循环size次,复制memFrom的值到memTo中
  7. *tempTo++ = *tempFrom++ ;
  8. return memTo;
  9. }

3. strcpy与memcpy的区别

3.1 复制的内容不同。

strcpy只能复制字符串,而memcpy可以复制任意内容,例如字符数组、整型、结构体、类等。

3.2 复制的方法不同。

strcpy不需要指定长度,它遇到字符串结束符"/0"便结束。memcpy则是根据其第3个参数决定复制的长度。

3.3 用途不同。

通常在复制字符串时用strcpy,而需要复制其他类型数据时则一般用memcpy。

4. strncpy

strncpy的实现如下

[c-sharp] view plain copy print ?
  1. char *strncpy(char *dst, const char *src, size_t count)
  2. {
  3. assert((dst != NULL) && (src != NULL));
  4. char *tmp = dst;
  5. while (count-- && (*dst++ = *src++) != '/0')
  6. {
  7. /* nothing */;
  8. }
  9. return tmp;
  10. }

该函数用于拷贝count个字符。

注意:

1. count一定要小于dst的大小。

2.调用完该函数后一定要加上一句:dst[count] = '/0';否则不安全,如strlen等函数要求参数必须是以'/0'结尾的字符串。

因为当count小于src的大小时,src的结束字符'/0'并不会被拷贝,故应该为dst加上一个结束符。

例:

[c-sharp] view plain copy print ?
  1. char *str = "123456";
  2. char dst[7];
  3. int count = 6;
  4. strncpy(dst, str, count); //count=6一定要小于dst的长度(7)。
  5. dst[count] = '/0'

转载地址:http://blog.csdn.net/gogor/article/details/4511430

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C/C++面试之算法系列--几个典型的内存拷贝及字符串函数实现 写一个函数,完成内存之间的拷贝。[考虑问题是否全面,是否考虑内存重叠问题] 返回void *支持链式操作,参数类型是void *以支持任意类型的指针,输入参数加上const修饰,最好加上assert对输入输出指针进行非NULL判断 void* memcpy( void *dest, const void *src, size_t count ) { char* pdest = static_cast<char*>( dest ); const char* psrc = static_cast<const char*>( src ); // 依次从前拷贝,目的地址覆盖了源地址的数,此时从后往前拷贝 if( (pdest>psrc) && (pdest<(psrc+count))) //能考虑到这种情况就行了 { for( size_t i=count-1; i!=-1; --i ) pdest[i] = psrc[i]; } else { for( size_t i=0; i<count; ++i ) pdest[i] = psrc[i]; } return dest; } int main( void ) { char str[] = "0123456789"; memcpy( str+1, str+0, 9 ); cout << str << endl; memcpy( str, str+5, 5 ); cout << str << endl; system( "Pause" ); return 0; // 0012345678 // 4567845678 } strcmp(): Int simple_strcmp (const char *s1, const char *s2) { int ret; while ((ret = *(unsigned char *) s1 - *(unsigned char *) s2++) == 0 && *s1++); return ret; } memcmp(): int simple_memcmp (const char *s1, const char *s2, size_t n) { int ret = 0; while (n--&& (ret = *(unsigned char *) s1++ - *(unsigned char *) s2++) == 0); return ret; } strcmp只判断s1的‘/0’,没有长度的控制;memcmp有长度参数n的控制 memcpy(): char *simple_memcpy (char *dst, const char *src, size_t n) { char *ret = dst; while (n--) *dst++ = *src++; return ret; } 直接的内存之间的copy,不处理内存重叠的情况。 strcpy(): char *simple_strcpy (char *dst, const char *src) { char *ret = dst; while ((*dst++ = *src++) != '\0'); //相当简约,一句话即搞定 return ret; } 与memcpy区别就是对'\0'作为结束符 strncpy(): char *simple_strncpy (char *dst, const char *src, size_t n) { char *ret = dst; while (n--) { if ((*dst++ = *src++) == '\0') { while (n--) *dst++ = '\0'; return ret; } } return ret; } n和‘\0'都作为结束的控制符。如果n过大,则后面全补'\0'

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值