当前位置: 首页 > news >正文

C语言内存函数(1)【memcpy函数的使用与模拟实现】【memmove函数的使用和模拟实现】

关于内存函数有四个函数需要我们学习。分别是memcpy,memmove,memset和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有任何的重叠,复制的结果都是未定义的。下面我来用代码演示一遍。

#include<stdio.h>
#include<string.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函数的模拟实现

这个函数的模拟实现不是太容易,我尽量写的详细一点。

#include<stdio.h>
#include<assert.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左边的时候咱们就从前往后,在右边的时候咱们就从后向前。这里我画一个区间来让大家更好理解。

这个都可以的意思就是在这里的位置已经不重叠了,不论从哪里开始都可以。现在我来用代码来实现一下这个代码。

#include<stdio.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(arr+2, arr, 20);for (int i = 0; i < 10; i++){printf("%d ", arr[i]);}return 0;
}

这两个函数的模拟实现跟之前的函数相比难度有些提升了。不过不影响我们去学会。

感谢大家的观看,如有错误请多多指正。

相关文章:

  • Fabric Measurement
  • 并发编程之Java 对象头的详细解析
  • 3个Tips,用“AI”开启新生活
  • 使用Python抓取抖音直播间数据的简易指南【第152篇—抓取数据】
  • 使用 PyOpenGL 进行 2D 图形渲染总结
  • 康奋威科技邀您到场参观2024长三角快递物流展
  • 阿里云原生:如何熟悉一个系统
  • Docker入门到实践之环境配置
  • 【算法每日一练]-图论(保姆级教程篇16 树的重心 树的直径)#树的直径 #会议 #医院设置
  • 【java数据结构】基于java提供的ArrayList实现的扑克牌游戏-(附源码~)
  • 上位机图像处理和嵌入式模块部署(qmacvisual图像识别)
  • 部署prometheus 监控k8s集群
  • Rust之构建命令行程序(五):环境变量
  • FANUC机器人零点标定的基本步骤(出厂数据)
  • PINN物理信息网络 | 全局自适应物理信息神经网络SA-PINN
  • [Vue CLI 3] 配置解析之 css.extract
  • [原]深入对比数据科学工具箱:Python和R 非结构化数据的结构化
  • 【108天】Java——《Head First Java》笔记(第1-4章)
  • 创建一个Struts2项目maven 方式
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 京东美团研发面经
  • 力扣(LeetCode)56
  • 小程序01:wepy框架整合iview webapp UI
  • 你对linux中grep命令知道多少?
  • k8s使用glusterfs实现动态持久化存储
  • # Java NIO(一)FileChannel
  • (MIT博士)林达华老师-概率模型与计算机视觉”
  • (差分)胡桃爱原石
  • (附源码)计算机毕业设计SSM保险客户管理系统
  • (离散数学)逻辑连接词
  • (一) springboot详细介绍
  • (转) ns2/nam与nam实现相关的文件
  • .NET CORE 第一节 创建基本的 asp.net core
  • .NET Entity FrameWork 总结 ,在项目中用处个人感觉不大。适合初级用用,不涉及到与数据库通信。
  • .NET 中使用 Mutex 进行跨越进程边界的同步
  • .net 逐行读取大文本文件_如何使用 Java 灵活读取 Excel 内容 ?
  • .Net下C#针对Excel开发控件汇总(ClosedXML,EPPlus,NPOI)
  • /bin/bash^M: bad interpreter: No such file ordirectory
  • ?php echo ?,?php echo Hello world!;?
  • @Autowired @Resource @Qualifier的区别
  • @ConditionalOnProperty注解使用说明
  • @media screen 针对不同移动设备
  • @在php中起什么作用?
  • [ IOS ] iOS-控制器View的创建和生命周期
  • [ 数据结构 - C++]红黑树RBTree
  • [AIGC] Java 和 Kotlin 的区别
  • [BZOJ 2142]礼物(扩展Lucas定理)
  • [C#]winform使用引导APSF和梯度自适应卷积增强夜间雾图像的可见性算法实现夜间雾霾图像的可见度增强
  • [c]统计数字
  • [C]整形提升(转载)
  • [caffe(二)]Python加载训练caffe模型并进行测试1
  • [ExtJS5学习笔记]第三十节 sencha extjs 5表格gridpanel分组汇总
  • [Flutter]设置应用包名、名称、版本号、最低支持版本、Icon、启动页以及环境判断、平台判断和打包
  • [Hadoop in China 2011] 蒋建平:探秘基于Hadoop的华为共有云
  • [html] 动态炫彩渐变背景