今天在使用memcpy的时候,突然发现有个函数的功能和memcpy函数功能是类似的,这个函数就是memmove。
于是我就很疑惑,这两个函数有啥区别呢,C语言标准函数库为啥要弄两个功能相识的函数呢。
看了下这篇博文memcpy与memmove的区别,知道了两个函数的区别:memmove在处理有内存重叠时会更加安全,但是这篇博文没有测试代码,因此我就接着这篇文章来写一些代码直观地测试一下。
当内存没有发生重叠时
#include <stdio.h>
#include <string.h>
int main() {
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int i;
printf("原数组: ");
for(i = 0; i < 10; i++) {
printf("%d ",arr[i]);
}
printf("\n内存操作之后:");
memcpy(arr, arr + 3, sizeof(int)*2);//memmove(arr, arr + 3, sizeof(int)*2)
for(i = 0; i < 10; i++) {
printf("%d ",arr[i]);
}
}
当内存发生重叠时
#include <stdio.h>
#include <string.h>
int main() {
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int i;
printf("原数组: ");
for(i = 0; i < 10; i++) {
printf("%d ",arr[i]);
}
printf("\n内存操作之后:");
memcpy(arr, arr + 3, sizeof(int)*5);//memmove(arr, arr + 3, sizeof(int)*5)
for(i = 0; i < 10; i++) {
printf("%d ",arr[i]);
}
}
我么用一张图来说明上面的情况
我们很容易知道,当原地址和目标地址没有发生内存重叠时,memcpy和memmove的效果是一样的。当原地址和目标地址有内存重叠时,那就存在一个内存复制顺序,我们知道内存复制的过程是一般读取一个字节到寄存器中进行操作,复制到目的地址,那是先从低地址开始还是高地址开始呢?
即是先从arr+3复制到arr,还是先从arr+7复制到arr+4呢?
因为存在内存重叠,所以两种不同的复制顺序会影响最后的结果。如下图所示
理论上是存在这样两种情况的。
实际上我们是不确定memcpy会采用哪种方法的。
而memmove是会以从低地址开始的。
总结:当我们确定我们的源地址和目标地址是没有重叠时,我们可以用memcpy,一般来说memcpy会更快一些。如果不能确定源地址和目标地址是否有重叠时,建议用memmove,这样会更安全。
实际上现在memcpy在源地址和目标地址是否有重叠时,结果也是正常的,只是这个结果会依赖编译器,我们不能让结果依赖编译器的处理。