美文网首页iOS开发笔记
memmove 和 memcpy的区别

memmove 和 memcpy的区别

作者: Hello小小酥 | 来源:发表于2019-01-26 16:58 被阅读29次

  在看Category的源码时碰见了两个C语言函数memmovememcpy,在string.h中我们可以看见它的定义:

void    *memcpy(void *__dst, const void *__src, size_t __n);
void    *memmove(void *__dst, const void *__src, size_t __len);

  从名称定义上看memcpy为拷贝,memmove为移动,但是不要被它的名字给蒙蔽了,其实这两个函数的作用都是拷贝一定长度的内存内容。他们两个唯一的区别是:当内存发生局部重叠时memmove函数能够保证拷贝结果的正确性,而memcpy则不能保证拷贝结果的正确性;当内存没有发生重叠的时候两个函数的结果是一样的。
下面我们来通过图示了解一下这两个函数的区别:

内存没有发生重叠

dst和src内存区域不重叠时.png

因为dstsrc没有发生内存区域重叠,因此可以直接将src拷贝到dst的后面,并不会发生任何错误。

内存发生重叠

内存发生重叠有两种情况:dst所在的区域在src区域前面、dst所在的区域在src区域的后面。
dst所在的区域在src区域前面时,memcpymemmove是可以完成拷贝的,结果也是正确的:

dst和src内存区域重叠时并且dst所在的区域在src区域前面时.png
dst所在区域在src前面时,将src拷贝到dst里面,只需要从src的头部一个一个拷贝到dst里面即可。
dst所在区域在src后面时,我们看一下图:
dst和src内存区域重叠时并且dst所在的区域在src区域后面时.png
此时如果还用memcpy函数的话,它会将src从头一个一个地拷贝到dst下面,这时如果从src的头部开始拷贝的话,会把内存地址为0x004里面的内容覆盖掉,这一部分是dst的开头但也是src的一部分,因此此种情况下memcpy函数会将重叠的部分覆盖掉;而在这种情况时memmove会选择从src的尾部开始拷贝(按箭头1、2、3、4),这样的话,重叠的内容就不会被覆盖,因此可以正确拷贝。我们在实际使用时如果在不确定内存地址是否重叠的情况下最好选择memmove函数,它能保证结果的正确性。

函数内部实现

memmove函数大体实现思路如下:

void *memmove(void *dest, const void *src, size_t count)
{
  char *ret = dst;
  if (src < dst)
    {
      dst += n;
      src += n;
      while (n--)
    *--dst = *--src;
    }
  else
    while (n--)
      *dst++ = *src++;
  return ret;
}

memcpy函数大体实现思路如下:

void *memcpy(void *s1, const void *s2, register size_t n)
{
  while (n--)
    *dst++ = *src++;
  return dst;
}

  以上代码为参考simple_memmove方法后得出的,具体的C标准库源码我们可以在这进行下载(打开网页后选择最新的glibc-x.x.x.tar.gz进行下载查看)。
  通过上述代码我们可以看到memmove函数的一个分支为memcpy函数的实现,而这一部分的实现就是内存地址没有发生重叠或者内存地址发生重叠并且dstsrc区域前面时,而当内存地址发生重叠并且dstsrc区域后面时memmove执行了if分支代码。

结束

  虽然memmove名称是移动,但是它和memcpy函数一样,都是执行的拷贝。而之所以会有memcpy函数的存在,是因为它的执行效率高,它里面没有像memmove函数里面的if分支,在我们能确定要拷贝的两个在内存地址没有重叠的情况下选择memcpy还是比较理想的。
  文章若有不足之处还请不吝赐教,大家互相学习。如果您觉得我的文章有用,点一下喜欢就可以了哦~~~。

相关文章

  • memmove 和 memcpy 的区别

    memmove和memcpy都是拷贝,区别是memmove会判断方向,如果有重叠的区域那么memcpy就会有问题 ...

  • memmove 和 memcpy的区别

    memcpy和memmove()都是C语言中的库函数,在头文件string.h中,作用是拷贝一定长度的内存的内容,...

  • memmove 和 memcpy的区别

      在看Category的源码时碰见了两个C语言函数memmove和memcpy,在string.h中我们可以看见...

  • memmove和memcpy

    内存移动void *memmove(void *str1, const void *str2, size_t n)...

  • memcpy与memmove的区别

    memcpy和memmove都是C语言的库函数,相比于strcpy和strncpy只能拷贝字符串数组,memcpy...

  • memset, memcpy and memmove

    memset 函数原型 头文件include 功能将dest所指的内存区域的前size个字节的值全部设置为ch指...

  • memcpy memmove 函数

    memcpy函数 函数原型 功能:由src指向地址为起始地址的连续 size个字节的数据复制到以dest指向地址为...

  • memmove与memcpy

    我们知道,分类的方法列表是通过memmove和memcpy加到类的方法列表的前面,而同名的分类方法根据文件的编译顺...

  • 不可忽视的memmove与memcpy区别

    同样的服务端程序,在ubunut下面运行正常,在centos7下面会运行出错。出错的原因在于在调用recv,返回值...

  • strcpy函数和memcpy

    strcpy和memcpy的区别

网友评论

    本文标题:memmove 和 memcpy的区别

    本文链接:https://www.haomeiwen.com/subject/cvgbjqtx.html