前言:
我们之前学过strcpy,strcmp等等函数,他们可以拷贝字符串和比较字符串等等,那么有没有什么函数不光可以拷贝字符串还可以拷贝其他的数据呢,答案就是内存函数。
相较于字符串函数,内存函数可以拷贝的数据类型不受限制,故名思意,内存函数就是直接操作内存块的,因此任何类型的数据都可以进行操作,那么接下来我们需要学习的内存函数有memcpy,memmove,memset,memcmp。
memcpy
memcpy函数是一个内存拷贝函数,它可以将一块内存的数据拷贝给另一块内存,以下是它的函数原型:
该函数存在三个参数:
- dest:指向目标内存,简单说就是我们需要拷贝的数据存放的对象,比如我们像把数组1的数据拷贝给数组2,那么数组2就是拷贝的目标
- src:指向拷贝的来源,就是我们数据拷贝的来源,借用上面的例子就是数组1
- num:需要我们传入拷贝的内存大小,以字节为单位,比如说我们要拷贝一个int类型,那么num就需要传入4,或者sizeof(int)。
函数使用代码展现:
// 内存函数
// memcpy
int main()
{// 来源数组1int arr1[] = { 0,1,2,3,4,5,6,7,8,9,10 };// 目标数组2int arr2[20] = { 0 };// 拷贝memcpy(arr2, arr1, 11 * sizeof(int));// 打印for (int i = 0; i < 20; i++){printf("%d ", arr2[i]);}return 0;
}
运行结果:
这边注意我们拷贝的目标数组的容量一定要大于或者等于拷贝数据的大小,否则会出现错误。
了解了使用方法后,让我们来自己实现这个函数的功能把。
函数分析:
通过函数模型知道我们也需要dst,src指针指向目标和来源,num传入拷贝的大小(以字节为单位)。
void* my_memcpy(void* dst, const void* src, size_t num)
{void* ret = dst;while (num--){*(char*)dst = *(char*)src; // 用char*进行强转保障每次的拷贝是一个字节// 建议写法1,不可以直接++// 写法1/*dst = (char*)dst + 1;src = (char*)src + 1;*/// 写法2((char*)dst)++;((char*)src)++;}return ret;
}
memmove
memmove函数的使用方法和功能和memcpy基本一致,唯一有区别的是,如果按照我们上面实现memcpy函数的方法,我们是无法实现内存重叠的拷贝的,什么是内存重叠呢,我们来看下面一段代码:
int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,10 };my_memcpy(arr+2, arr, 5 * sizeof(int));for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}
我们自己拷贝给自己,arr代表数组首元素地址,arr+2指向的就是3,我们需要将1,2,3,4,5,从3的位置拷贝上去。
最后的拷贝如上图,但是实际我们运行时如下图。
因为拷贝的时候都是从头到尾,会发生覆盖问题,为了解决这样内存块重叠拷贝的问题,推出了memmove函数,当然现在在VS上的memcpy也可以解决这样的问题,但是涉及内存块重叠拷贝还是建议使用memmove函数。
函数原型:
该函数存在三个参数:
- dest:指向目标内存,简单说就是我们需要拷贝的数据存放的对象,比如我们像把数组1的数据拷贝给数组2,那么数组2就是拷贝的目标
- src:指向拷贝的来源,就是我们数据拷贝的来源,借用上面的例子就是数组1
- num:需要我们传入拷贝的内存大小,以字节为单位,比如说我们要拷贝一个int类型,那么num就需要传入4,或者sizeof(int)。
函数使用代码展现:
int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,10 };memmove(arr + 2, arr, 5 * sizeof(int));for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}
运行结果:
代码实现:
void* my_memmove(void* dst, const void* src, size_t num)
{void* ret = dst;if (dst < src){//前->后while (num--){*(char*)dst = *(char*)src;dst = (char*)dst + 1;src = (char*)src + 1;}}else{// 后->前while (num--){*((char*)dst + num) = *((char*)src + num);}}return ret;
}int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,10 };my_memmove(arr+4, arr+2, 5 * sizeof(int));for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}
memset
memset是⽤来设置内存的,将内存中的值以字节为单位设置成想要的内容。
int main ()
{
char str[] = "hello world";
memset (str,'x',6);
printf(str);
return 0;
}
因为是以每一个字节取设置内存,对于其他存储高于一个字节的类型只能初始化,不可以设置其他的值,会产生错误:
int main()
{int arr[10];memset(arr, 0, 10*sizeof(int));for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}
int main()
{int arr[10];memset(arr, 1, 10*sizeof(int));for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}
memcmp
⽐较从ptr1和ptr2指针指向的位置开始,向后的num个字节,内存函数的比较也是一个字节一个字节的比较。
函数原型:
#include <stdio.h>
#include <string.h>
int main()
{
char buffer1[] = "DWgaOtP12df0";
char buffer2[] = "DWGAOTP12DF0";
int n;
n = memcmp(buffer1, buffer2, sizeof(buffer1));
if (n > 0)
printf("'%s' is greater than '%s'.\n", buffer1, buffer2);
else if (n < 0)
printf("'%s' is less than '%s'.\n", buffer1, buffer2);
else
printf("'%s' is the same as '%s'.\n", buffer1, buffer2);
return 0;
}