memmove 和 memcpy的区别

  在看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则不能保证拷贝结果的正确性;当内存没有发生重叠的时候两个函数的结果是一样的。
下面我们来通过图示了解一下这两个函数的区别:

内存没有发生重叠

memmove 和 memcpy的区别_第1张图片
dst和src内存区域不重叠时.png

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

内存发生重叠

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

memmove 和 memcpy的区别_第2张图片
dst和src内存区域重叠时并且dst所在的区域在src区域前面时.png

dst所在区域在src前面时,将src拷贝到dst里面,只需要从src的头部一个一个拷贝到dst里面即可。
dst所在区域在src后面时,我们看一下图:
memmove 和 memcpy的区别_第3张图片
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的区别)