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

2024-03-24 09:44

本文主要是介绍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;
}

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

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

这篇关于C语言内存函数(1)【memcpy函数的使用与模拟实现】【memmove函数的使用和模拟实现】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/841225

相关文章

Golang基于内存的键值存储缓存库go-cache

《Golang基于内存的键值存储缓存库go-cache》go-cache是一个内存中的key:valuestore/cache库,适用于单机应用程序,本文主要介绍了Golang基于内存的键值存储缓存库... 目录文档安装方法示例1示例2使用注意点优点缺点go-cache 和 Redis 缓存对比1)功能特性

Python如何实现读取csv文件时忽略文件的编码格式

《Python如何实现读取csv文件时忽略文件的编码格式》我们再日常读取csv文件的时候经常会发现csv文件的格式有多种,所以这篇文章为大家介绍了Python如何实现读取csv文件时忽略文件的编码格式... 目录1、背景介绍2、库的安装3、核心代码4、完整代码1、背景介绍我们再日常读取csv文件的时候经常

Golang中map缩容的实现

《Golang中map缩容的实现》本文主要介绍了Go语言中map的扩缩容机制,包括grow和hashGrow方法的处理,具有一定的参考价值,感兴趣的可以了解一下... 目录基本分析带来的隐患为什么不支持缩容基本分析在 Go 底层源码 src/runtime/map.go 中,扩缩容的处理方法是 grow

Go 1.23中Timer无buffer的实现方式详解

《Go1.23中Timer无buffer的实现方式详解》在Go1.23中,Timer的实现通常是通过time包提供的time.Timer类型来实现的,本文主要介绍了Go1.23中Timer无buff... 目录Timer 的基本实现无缓冲区的实现自定义无缓冲 Timer 实现更复杂的 Timer 实现总结在

基于Python实现多语言朗读与单词选择测验

《基于Python实现多语言朗读与单词选择测验》在数字化教育日益普及的今天,开发一款能够支持多语言朗读和单词选择测验的程序,对于语言学习者来说无疑是一个巨大的福音,下面我们就来用Python实现一个这... 目录一、项目概述二、环境准备三、实现朗读功能四、实现单词选择测验五、创建图形用户界面六、运行程序七、

如何使用Docker部署FTP和Nginx并通过HTTP访问FTP里的文件

《如何使用Docker部署FTP和Nginx并通过HTTP访问FTP里的文件》本文介绍了如何使用Docker部署FTP服务器和Nginx,并通过HTTP访问FTP中的文件,通过将FTP数据目录挂载到N... 目录docker部署FTP和Nginx并通过HTTP访问FTP里的文件1. 部署 FTP 服务器 (

Vue中动态权限到按钮的完整实现方案详解

《Vue中动态权限到按钮的完整实现方案详解》这篇文章主要为大家详细介绍了Vue如何在现有方案的基础上加入对路由的增、删、改、查权限控制,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、数据库设计扩展1.1 修改路由表(routes)1.2 修改角色与路由权限表(role_routes)二、后端接口设计

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

C#集成DeepSeek模型实现AI私有化的流程步骤(本地部署与API调用教程)

《C#集成DeepSeek模型实现AI私有化的流程步骤(本地部署与API调用教程)》本文主要介绍了C#集成DeepSeek模型实现AI私有化的方法,包括搭建基础环境,如安装Ollama和下载DeepS... 目录前言搭建基础环境1、安装 Ollama2、下载 DeepSeek R1 模型客户端 ChatBo

Qt实现发送HTTP请求的示例详解

《Qt实现发送HTTP请求的示例详解》这篇文章主要为大家详细介绍了如何通过Qt实现发送HTTP请求,文中的示例代码讲解详细,具有一定的借鉴价值,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1、添加network模块2、包含改头文件3、创建网络访问管理器4、创建接口5、创建网络请求对象6、创建一个回复对