小企业网站建设费用一年,公司没有网站如何做外贸,南阳专业做网站公司,新新手手网网站站建建设设关于内存函数有四个函数需要我们学习。分别是memcpy#xff0c;memmove#xff0c;memset和memcmp。都在头文件string.h里面。
一.memcpy函数的使用
一提到这个函数#xff0c;我们可能会联想到strcpy函数#xff0c;但strcpy函数是针对字符串的拷贝。但是我们在写代码的…关于内存函数有四个函数需要我们学习。分别是memcpymemmovememset和memcmp。都在头文件string.h里面。
一.memcpy函数的使用
一提到这个函数我们可能会联想到strcpy函数但strcpy函数是针对字符串的拷贝。但是我们在写代码的时候不可能只拷贝字符串。 int arr1[] { 1,2,3,4,5,6,7,8,9,0 };int arr2[20] { 0 };
在这里我想把arr1前五个元素拷贝到arr2里面要怎么样实现呢这里我们就可以使用memcpy。大家注意memcpy是针对内存块进行拷贝的。它是有三个参数的。 它的前两个参数都是void*类型的指针。
1函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存位置2这个函数在遇到\0后不会停下来3如果source和destination有任何的重叠复制的结果都是未定义的。下面我来用代码演示一遍。
#includestdio.h
#includestring.h
int main()
{int arr1[] { 1,2,3,4,5,6,7,8,9,0 };int arr2[20] { 0 };memcpy(arr2, arr1, 20);//注意这里的20单位是字节for (int i 0; i 5; i){printf(%d , arr2[i]);}return 0;
}
最终也是成功可以打印出来1,2,3,4,5。到这里相信也可以看出来这个函数具体的作用了。
二.memcpy函数的模拟实现
这个函数的模拟实现不是太容易我尽量写的详细一点。
#includestdio.h
#includeassert.h
void* my_memcpy(void* dest, const void* sec, size_t num)
{void* ret dest;//先把dest的起始地址给存起来assert(dest sec);//assert断言判断dest和sec是不是空指针while (num--)//num总共有20个字节这里循环20次{*(char*)dest *(char*)sec;//大家注意void*类型的指针不能直接解引用这里强制转换成char*类型的指针//强制类型转换后再次解引用也就是使用char*类型解引用访问一个字节所以这里赋值的时候也是一次赋值一个字节//至于为什么强制转换成char*类型是为了避免num为单数比如3,5,7。((char*)sec);//注意也不要写成char*sec因为这里强转只是临时的当我们的时候就不是强转之后的结果。((char*)dest);//这里也是一次加一个字节往后赋值}return ret;
}
int main()
{int arr1[] { 1,2,3,4,5,6,7,8,9,0 };int arr2[20] { 0 };my_memcpy(arr2, arr1, 20);//注意这里的20单位是字节for (int i 0; i 5; i){printf(%d , arr2[i]);}return 0;
}
大家注意假如我的内存有重叠的话这个我写的memcpy函数的模拟实现是实现不了的。简单的说我想把arr1的前五个元素拷贝到3,4,5,6,7的位置上将arr1数组的元素变成1,2,1,2,3,4,5,8,9,0.这里我截屏看一下。
大家可以看到如果我用我写的my_memcpy函数是不能实现的。因为当我们把数组的第三个元素给覆盖之后这里就变成了1第四个元素就变成了2。当我们再次把第三个元素赋值给第五个元素的时候其实是把已经变成1的第三个元素赋值给第五个元素。依次类推赋值的结果就只能是一开始就复制成功的1和2。
但是如果我不使用自己写的我们使用库函数的话大家再来看。
这里竟然也成功的赋值了。还记得上面我说的第三个定义吗即使这个库函数可以实现这样内存重叠的赋值但是我们不给予这个函数可以达到这个作用的希望。我们认为memcpy这个函数只要可以完成内存没有重叠时赋值的作用就可以了。对于内存重叠的部分我们用另一个库函数来进行。上面我写的my_memcpy就已经可以发挥出memcpy的作用了。接下来我就来介绍memmove。
三.memmove函数的使用
这个函数的参数和memcpy的参数都是一样的。 memmove与memcpy差别就是memmove处理的源内存块和目标内存块是可以重叠的。而且只要源内存块和目标内存块只要出现重叠就得使用memmove函数来处理。
还是用上面的代码来让大家看一下。 这里的使用我就不过多介绍了因为这个函数使用起来是比较简单的难的是模拟实现。
四.memmove的函数模拟实现
关于这个函数的实现我们主要考虑的就是要避免值被覆盖。这里有几种情况大家看一下。 还有从前往后赋值的情况。假如我要把3,4,5,6,7,赋值给1,2,3,4,5。这时就不能再去使用这种从前往后的方式了。 也就是说当dest在src左边的时候咱们就从前往后在右边的时候咱们就从后向前。这里我画一个区间来让大家更好理解。
这个都可以的意思就是在这里的位置已经不重叠了不论从哪里开始都可以。现在我来用代码来实现一下这个代码。
#includestdio.h
#include assert.h
void* my_memmove(char* dest, const char* src, size_t num)
{assert(dest src);//assert判断空指针的if (dest src)//前-后{while (num--){*(char*)dest *(char*)src;dest (char*)dest 1;src (char*)src 1;}}else//后-前{while (num--){*((char*)dest num) *((char*)src num);}}
}
int main()
{int arr[] { 1,2,3,4,5,6,7,8,9,0 };my_memmove(arr2, arr, 20);for (int i 0; i 10; i){printf(%d , arr[i]);}return 0;
}
这两个函数的模拟实现跟之前的函数相比难度有些提升了。不过不影响我们去学会。
感谢大家的观看如有错误请多多指正。