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

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置