C语言中,除了strlen,strcpy,strcmp等一系列字符串函数之外,还有与之类似的有关内存的函数memcpy,memmove等,这些函数与字符串函数最大的区别就是字符串的函数形参只能是char*,而内存相关的函数可以对各种类型的数据进行拷贝、修改等操作。
一、memcpy
将source后面num个字节的数据拷贝到destination的位置,与strcpy类似,但可以设置拷贝的长度。
自主实现
void memcpy(void* dest, const void* src, size_t sz)
{
while (sz--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
}
int main()
{
char arr1[] = "abcdegg";
char arr2[] = "fedd";
memcpy(arr1, arr2, sizeof(arr2));
printf("%s\n", arr1);
return 0;
}
二、memmove
与memcpy类似,memmove同样是内存拷贝函数。
在使用memcpy包括使用strcpy时,可能会出现以下问题:
这段代码我们想要将12345拷贝arr[2]后面即原数组34567的位置,但没有拷贝成功,这是因为我们在拷贝第三次时,arr[2]的数据已经被改成1了,3已经丢失,无法找到原数组12345中的345了。
因此在拷贝一段重叠的内存时,不能使用memcpy,而是要使用memmove。
那么如何自己实现memmove函数呢?
我们发现,在拷贝内存时,如果目标位置在在源位置的前面,即dest<src,那么就可以用memcpy进行拷贝,不会发生重叠;当dest>=src,这时候再使用memcpy的方式就会发生重叠,而如果我们改变顺序,从地址的最后进行倒序拷贝,就会解决重叠的问题。
void My_memmove(void* dest, void* src, size_t sz)
{
if (dest < src)
{
while (sz--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
}
else
{
while (sz--)
{
*((char*)dest + sz) = *((char*)src + sz);
}
}
}
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
My_memmove(&arr[2], arr, 20);
for (int i = 0; i < 10; i++)
{
printf("%d ", arr[i]);
}
return 0;
}