???学习的道路很枯燥,希望我们能并肩走下来!
编程真是一件很奇妙的东西。你只是浅尝辄止,那么只会觉得枯燥乏味,像对待任务似的应付它。但你如果深入探索,就会发现其中的奇妙,了解许多所不知道的原理。知识的力量让你沉醉,甘愿深陷其中并发现宝藏。
前言
本篇是详细介绍内存函数的模拟实现,让使用者对内存函数有进一步认识,而不是仅仅停留在表面,更好的模拟,为了更好的使用. 文章可能出现错误,如有请在评论区指正,让我们一起交流,共同进步!
本文开始
1.memcpy:内存拷贝
memcpy作用:将数组的字节数的值,从源头指向的位置直接复制到目标所指向的位置.(数组是void 类型可接受任何类型,例如:字符数组,整型数组等) destination(目的地): 指向要在其中复制内容的目标数组的指针 source:指向要复制的源头数组的指针 size_t:无符号整型,存储要复制的字节数 void :不在局限于只能接收整形的单一类型,可以接收多种类型字符类型等
模拟实现内存拷贝代码图示:
代码my-memcpy解释: 总共n个字节,肯定需要拷贝n次,所以使用循环,再将从源头指向的字节,拷贝到目标指向的空间,拷贝一个我们地址+1,所以我们使用,先将目标(源头)地址转换为char,再+1传递给目标(源头);*
为什么不使用(char*)dest++或者(char*)src++? 其实强制转换只是临时的,而++是后置的,当再操作dest时已经不是char类型了,所以我们只能使用(char)dest + 1;
为什么要使用char* 类型? 注释:因为要接收不同的类型使用void,不论是整形,还是浮点型等类型,都可以将他们算出需要拷贝的字节数,我们就可以使用字节传递,而不需要考虑他是什么类型了,把传递的类型改为char 一个一个字节传递,就可以实现他们的拷贝.
2.memmove:内存移动
const void * source :const修饰source让source的值不变 作用:这个内存函数与参数memcpy类似但是使用不同,memove:可以重叠使用;memcpy:不能重叠使用;
重叠使用就是在一个数组中将一段数字,复制到该数组中的任意一段并有重叠部分 例如下图:将绿色方框的数字移动到蓝色方框或者橙色方框
对于memmove移动有三种情况: 重叠: 从绿框移动到蓝框,前到后复制;src=>dest 从绿框移动到橙框,后到前复制;src+n=>dest 不重叠:可以从前到后,也可以从后到前复制
模拟memmove代码图示:
3.memcmp:内存比较
作用:将ptr1所指向的比较的字节(比较的数组1),ptr2所指向的比较字节(比较的数组2),一个一个字节比较,直到字节结束或者出现不同字符;如果一样返回0,否则返回非0值; num: 所比较的字节数
memcmp代码图示: 相同=》下标+1,不同相减》返回非0值
4.memset:内存设置
作用:设置指针指向的内容;设置需要一个一个字节设置,value是整形需要强转为char来设置 ptr: 指向所要设置值的内存空间 value: 想要设置的值 num: 设置的字节数
5.函数代码
5.1memmove代码
void* my_memove(void* dest, const void* src, size_t n)
{
void* ret = dest;
//重叠看dest 与 src 的前后位置决定
if (dest < src)
{
while (n--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
}
else//包含重叠和不重叠的情况
{
while (n--)
{
*((char*)dest + n) = *((char*)src + n);
}
}
return ret;
}
int main()
{
int arr[] = { 1,2,3,4,5,6,7,8,9,0 };
my_memove(arr + 2, arr,20);//把1,2,3,4放到3,4,5,6
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr[i]);
}
return 0;
}
5.2memcpy代码
void* my_memcpy(void* dest, void* src, size_t n)
{
void* ret = dest;
while (n--)
{
*(char*)dest = *(char*)src;
dest = (char*)dest + 1;
src = (char*)src + 1;
}
return ret;
}
int main()
{
int arr[] = { 1,2,3,4,5,6,7,8,9,0 };//3,4,5,6,7,6,7,8,9,0
my_memcpy(arr, arr+2, 20);
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", arr[i]);
}
return 0;
}
5.3memcmp代码
#include<string.h>
int my_memcmp(const void* p1, const void* p2, size_t n)
{
int i = 0;
for (i = 0; i < n; i++)
{
if (*((char*)p1) == *((char*)p2))//相同下标+1
{
p1 = (char*)p1 + 1;
p2 = (char*)p2 + 1;
}
else
{
return *((char*)p1) - *((char*)p2);//不同值相减
}
}
return 0;//遍历完没有不同的,返回0 =》相等
}
int main()
{
char b1[] = "DWga";
char b2[] = "DWGA";
int n;
n = my_memcmp(b1, b2, sizeof(b1));
if (n > 0)
{
printf("b1 > b2\n");
}
else if (n < 0)
{
printf("b1 < b2\n");
}
else
{
printf("b1 = b2\n");
}
return 0;
}
5.4memset代码
void* my_memset(void* ptr, int value, size_t num)
{
void* ret = ptr;
int i = 0;
while(num--)
{
*((char*)ptr) = (char)value;
ptr = (char*)ptr + 1;
}
return ret;
}
int main()
{
int arr[] = {1,2,3,4,5};
my_memset(arr,0, 8);
int i = 0;
for (i = 0; i < 5; i++)
{
printf("%d ", arr[i]);
}
return 0;
}
总结
???各位读友,本篇分享到内容是否更好的让你理解了内存函数,如果对你有帮助给个👍赞鼓励一下吧!! 🎉🎉🎉世上没有绝望的处境,只有对处境绝望的人。 感谢每一位一起走到这的伙伴,我们可以一起交流进步!!!一起加油吧!!! >
|