【C语言】memmove()函数(拷贝重叠内存块函数详解)

news2024/11/18 6:48:51

🦄个人主页:修修修也

🎏所属专栏:C语言

⚙️操作环境:Visual Studio 2022


目录

 一.memmove()函数简介

1.函数功能

2.函数参数

1>.void * destination

2>.onst void * source

3>.size_t num

3.函数返回值

4.函数头文件

二.memmove()函数的具体使用

1.使用memmove()函数完成拷贝整型数组数据(目的地与源重叠)

2.使用memmove()函数完成拷贝字符数组数据(目的地与源重叠)

三.模拟实现memmove()函数功能

🎏实现思路

1.函数参数及返回值设定逻辑

📌函数参数:

📌函数返回值:

2.函数功能实现逻辑

1.情况一(两个内存块不重叠)

2.情况二(两个内存块有重叠)

🎏代码编写

🎏运行测试

1.测试my_memmove()函数的逆序拷贝

2.测试my_memmove()函数的顺序拷贝

结语


 一.memmove()函数简介

我们先来看一下cplusplus.com - The C++ Resources Network网站上memmove()函数基本信息

1.函数功能

可以看到,memmove()函数的功能是:

源头指向的内存块拷贝固定字节数的数据目标指向的内存块,并且源头的内存块与目标内存块可以重叠.(最后一点是memmove()与memcpy最大的区别)


2.函数参数

该函数一共有三个参数,分别是:

void * memmove ( void * destination, const void * source, size_t num );

1>.void * destination

第一个参数的类型是无类型指针(void*),它指向拷贝的目的地内存块,它的作用是为函数提供目的地的内存块起始地址,以便函数能够准确地将内容拷贝到我们需要的内存空间.

2>.onst void * source

第二个参数的类型是被const修饰(const修饰的指针,const在*左边表示指针指向的内容不可修改,const在*右边表示指针的指向不可修改)无类型指针(void*),它指向拷贝数据的来源内存块,它的作用是为函数提供拷贝源头内存块起始地址,以便函数能够准确找到拷贝的源头进行拷贝.

3>.size_t num

第三个参数的类型是size_t(无符号整形),它表示要拷贝数据的字节数,它的作用是告诉函数需要拷贝的字节数是多少,以便函数精准的拷贝该数目字节数空间的内容到目的地.


3.函数返回值

函数的返回值类型是无类型指针(void*),它的作用是在函数运行结束后返回拷贝后的目的地内存块的起始地址.


4.函数头文件

该函数包含在头文件<string.h>中.


二.memmove()函数的具体使用

memmove()函数的使用场景是:

当我们想拷贝一个整型数组/结构体/枚举常量等strcpy()函数无法拷贝的数据,并且目的地内存块和源头内存块可能会有重叠的时候,我们可以考虑使用memmove()函数来完实现这一诉求,当然,想要使用memmove()函数拷贝字符串数据或者拷贝目的地内存块和源头内存块不重叠也是可以的.(但是会有些杀鸡用牛刀的感觉哈哈哈)

下面是拷贝时源内存块与目标内存块重叠的情况示意图:

1.使用memmove()函数完成拷贝整型数组数据(目的地与源重叠)

因为拷贝目的地内存块与源内存块不重叠的情况我们已经在memcpy()函数部分详细展示过了,因此在memmove()函数部分我们将着重展示它的内存块重叠时的使用情况.

如下,我们使用memmove()函数将arr数组中的1,2,3,4,5拷贝到3,4,5,6,7的位置上去:

分别给memmove()函数传入三个参数:

拷贝目的地地址(即arr+2),拷贝来源地址(即arr),拷贝字节数(即sizeof(arr[0])*5).

/* memmove 使用测试 */
#include <stdio.h>
#include <string.h>

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	memmove(arr+2, arr, sizeof(arr[0])*5);		
	//sizeof(arr[0])计算的结果是arr数组中一个元素的字节大小,乘5代表5个
	
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

在vs编译器中运行查看结果:

可以看到memmove()函数成功的将arr数组中的1,2,3,4,5拷贝到了3,4,5,6,7的位置上.


2.使用memmove()函数完成拷贝字符数组数据(目的地与源重叠)

如下,我们使用memmove()函数将str数组的" very useful"拷贝到" useful......"的位置上去:

分别给memmove()函数传入三个参数:

拷贝目的地地址(即str+20),拷贝来源地址(即str+15),拷贝字节数(即11).

/* memmove 使用测试 */
#include <stdio.h>
#include <string.h>

int main()
{
	char str[] = "memmove can be very useful......";
	memmove(str + 20, str + 15, 11);
	//使用memmove()函数拷贝内存块重叠字符串

	printf("%s\n", str);
	return 0;
}

在vs编译器中运行查看结果:

 可以看到memmove()函数成功的将str数组的" very useful"拷贝到了" useful......"的位置上.


三.模拟实现memmove()函数功能

🎏实现思路

1.函数参数及返回值设定逻辑

📌函数参数:
void * destination

因为memmove()函数要实现的是内存空间的拷贝,所以在使用memmove()函数时我们难免会遇到拷贝不同类型数据的可能,因此在这里我们需要将目的地的地址类型设置为无类型指针(void*),以便函数后续可以处理任意类型的数据.

const void * source

来源地址的类型设置为无类型指针(void*)的原因与目的地的原因相同,都是便于函数可以处理任意类型的数据.

而给来源的地址指针加上const的原因是防止拷贝的过程中将来源的内容不慎修改,在*指针左侧加上const就可以使const修饰的指针指向的内容变成常量.

size_t num

 因为要拷贝的字节数恒为非负数,因此字节数的类型是无符号整形(size_t).

📌函数返回值:
void*

函数返回值设置为void*的原因同目的地及来源地相同,都是便于函数可以在处理完任意类型的数据后可以返回目的地的地址.


2.函数功能实现逻辑

在讲实现逻辑之前,我们先分情况讨论一下在拷贝数据时,我们可能遇到的几种情况:

注:以下演示中,*src指针指向源头起始内存块,*dest指针指向目标起始内存块.

1.情况一(两个内存块不重叠)

这种情况下内存空间的拷贝逻辑是较为简单的,不论是数据从前向后拷贝还是从后向前拷贝,结果都是正确的.

例如这种情况:

 或是这种情况:


2.情况二(两个内存块有重叠)

这种情况就比较复杂了,同样分两种情况,如图:

源头指针在前时,只能从源头内存块的后端向前端移动拷贝:

 

拷贝逻辑示意图(序号代表拷贝次序):

目的地指针在前时,只能从源头内存块的前端向后端移动拷贝:

拷贝逻辑示意图(序号代表拷贝次序):

分析清楚上面四种不同的情况,我们就可以在代码的编写中对不同情况进行分情况讨论了.

当然这里的分情况是十分灵活的,你可以四种情况各分一种,也可以按拷贝模式来分,从前向后拷的算一种,从后向前拷的算另一种.

除了两个必须按固定方式拷贝的情况之外,剩下两种情况无论按哪种方式拷贝都行.

那么我们在这里就选择一种最简单也最容易理解的方式来模拟实现memmove()函数吧.

如果源头指针在前(小),从后向前拷贝;目的地指针在前(小),从前向后拷贝

数据在内存中的存储示意图:

 


🎏代码编写

清楚了不同情况的拷贝逻辑后,代码的编写就只需要按上面说的分两种情况就好了,以及,函数在实现时的其他需要注意的点我都标注在注释中了.

综上,my_memmove()函数的完整实现代码如下:

//模拟实现my_memmove()函数

void* my_memmove(void* destination, const void* source, size_t num)
{
	assert(destination);        //防止源头或目的地指针为NULL
	assert(source);
	void* ret = destination;
	if (source < destination)   //内存中数据的存储是由低地址到高地址的
	{
		//从后向前拷贝
		while (num--)   //以num为20为例,在num进入while循环之后,就立刻--,变成19了
		{
			*((char*)destination + num ) = *((char*)source + num );
			//这时source+num刚好指向的是源头内存块的最后一个字节
		}
	}
	else
	{
		while (num--)
		{
			//从前向后拷贝
			*((char*)destination) = *((char*)source);
			++((char*)destination);        
			++((char*)source);
			//这里使用后置++的话一定要给(char*)destination整体带上括号
			//否则后置++的优先级比(char*)的强制类型转换的优先级高,
			//导致指针类型还是void*时就进行++操作,这是在C标准中是不允许的
		}
	}
	return ret;
}

🎏运行测试

1.测试my_memmove()函数的逆序拷贝

如下,我们使用my_memmove()函数将arr数组的1,2,3,4,5拷贝到3,4,5,6,7的位置上:

即按照前面提到的这种情况进行拷贝:

vs2022测试运行结果:

可以看到my_memmove()函数成功的将arr数组中的1,2,3,4,5拷贝到了3,4,5,6,7的位置上.


2.测试my_memmove()函数的顺序拷贝

如下,我们使用my_memmove()函数将arr数组的3,4,5,6,7拷贝到1,2,3,4,5的位置上:

即按照前面提到的这种情况拷贝:

vs2022中测试运行结果:

可以看到my_memmove()函数成功的将arr数组中的3,4,5,6,7拷贝到了1,2,3,4,5的位置上.


结语

希望这篇memmove()函数的介绍到能对大家有所帮助,欢迎大佬们留言或私信与我交流.

最后的最后,感谢这位大佬指出了我在memcpy()函数阶段模拟实现的不足,因为和他的交流,才促成了这篇博客的产生:

学海漫浩浩,我亦苦作舟!关注我,大家一起学习,一起进步!

相关文章推荐

【C语言】memcpy()函数

【C语言】memset()函数

【C语言】strcpy()函数

【C语言】strlen()函数

【C语言】rand()函数(如何生成指定范围随机数)

【实用调试技巧】总是找不到Bug?手把手教你在vs2022中调试程序



C语言内存相关库函数思维导图:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1142255.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【爬虫】charles手机抓包环境设置(设置系统证书)

1.说明 想要对手机抓包&#xff0c;最关键的是需要设置好根证书&#xff0c;用户证书在安卓7.0之后就不受信任了&#xff0c;想要对手机app抓包&#xff0c;就需要把用户证书设置为系统证书&#xff08;根证书&#xff09; 注意&#xff0c;想要设置为根证书&#xff0c;你的…

JavaScrip的DOM接口

JavaScript的DOM&#xff08;Document Object Model&#xff09;是一种接口&#xff0c;它允许程序和脚本动态地访问和更新文档的内容、结构和样式。DOM是一种将HTML或XML文档表示为对象树的标准方式。 在JavaScript中&#xff0c;DOM提供了一种方式来操作HTML或XML文档的元素…

基于nodejs+vue人脸识别考勤管理系统的设计与实现

目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性&#xff1a;…

unocss和tailwindcss css原子引擎

第一种tailwindcss&#xff1a; tailwindcss官网 https://tailwindcss.com/docs/grid-column 基本介绍及优点分析 Tailwind CSS 中文文档 - 无需离开您的HTML&#xff0c;即可快速建立现代网站 PostCss 处理 Tailwind Css 基本流程 PostCSS - 是一个用 JavaScript 工具和插…

分享Keil5设置忽略编译过程中的警告

Keil5忽略编译过程中的警告 分享Keil5设置忽略编译过程中的警告 分享Keil5设置忽略编译过程中的警告 加上这段字符就好了 --diag_suppress68 --diag_suppress111 --diag_suppress188 --diag_suppress223 --diag_suppress546 --diag_suppress1295实测有效

Vue3-02_Vue基础入门

背景 这里&#xff0c;跟vue官网的介绍章节稍有差异。官网上侧重组件原理&#xff0c;从浅到深介绍各种组件。后续是系统生态。 教程上更偏路线化&#xff0c;需要用到的优先讲解。完成综合案例。所以我主要按照教程的思路来进行学习。 ◆ 能够知道 vue 的基本使用步骤 ◆ 掌…

知识图谱相关的操作

微软生成自己的图谱&#xff1a;GitHub - microsoft/SmartKG: This project accepts excel files as input which contains the description of a Knowledge Graph (Vertexes and Edges) and convert it into an in-memory Graph Store. This project implements APIs to searc…

SSH安全登录远程主机

SSH服务器简介 SSH即Security SHell的意思&#xff0c;它可以将连线的封包进行加密技术&#xff0c;之后进行传输&#xff0c;因此相当的安全。 SSH是一种协议标准&#xff0c;其目的是实现安全远程登录以及其它安全网络服务。 SSH协定&#xff0c;在预设的状态下&#xff0c;…

计算机网络重点概念整理-第一章 计算机网络概述【期末复习|考研复习】

第一章 计算机网络概述 【期末复习|考研复习】 计算机网络系列文章传送门&#xff1a; 第一章 计算机网络概述 第二章 物理层 第三章 数据链路层 第四章 网络层 第五章 传输层 第六章 应用层 第七章 网络安全 计算机网络整理-简称&缩写 文章目录 第一章 计算机网络概述 【…

微信小程序之投票管理

前言 对于会议管理模块&#xff0c;必不可少的当然就是我们的投票管理&#xff0c;实现真正意义上的无纸化办公&#xff0c;本期博客为大家介绍会议管理模块&#xff0c;包括发布投票及查看各类投票的状态 所用技术点 MyBatis、SpringMVC、VentUI MyBatis和SpringMVC在博客主…

PyCharm社区版安装

PyCharm社区版安装 到中国官网下载 https://www.jetbrains.com/zh-cn/pycharm/download/?sectionwindows 首次创建项目&#xff0c;会自动下载安装Python 3.9 社区版的区别 社区版的区别

Vue项目搭建及使用vue-cli创建项目、创建登录页面、与后台进行交互,以及安装和使用axios、qs和vue-axios

目录 1. 搭建项目 1.1 使用vue-cli创建项目 1.2 通过npm安装element-ui 1.3 导入组件 2 创建登录页面 2.1 创建登录组件 2.2 引入css&#xff08;css.txt&#xff09; 2.3 配置路由 2.5 运行效果 3. 后台交互 3.1 引入axios 3.2 axios/qs/vue-axios安装与使用 3.2…

【微信小程序】发布投票与用户投票完整讲解

目录 前言 组件功能示例 一、数据库 二、后端接口定义 三、前端准备 3.1 定义连接接口 3.2 Vant Weapp UI 组件库 3.3 授权登录与相关工具 四、小程序编写 4.1 投票组件 WXML WXSS JSON WXJS 效果展示讲解&#xff1a; 4.2 发布投票组件 WXML WXSS JSON WX…

线扫相机DALSA--采集卡Base模式设置

采集卡默认加载“1 X Full Camera Link”固件&#xff0c;Base模式首先要将固件更新为“2 X Base Camera Link”。 右键SCI图标&#xff0c;选择“打开文件所在的位置”&#xff0c;找到并打开SciDalsaConfig的Demo&#xff0c;如上图所示&#xff1a; 左键单击“获取相机”&a…

生成的二维码如何解析出原来的地址?

生成的二维码如何解析出原来的地址&#xff1f; 随着移动互联网的发展&#xff0c;二维码作为一种快速获取信息的方式&#xff0c;在我们的生活中越来越常见。而PHP作为Web语言之一&#xff0c;也有着二维码解码的功能。 PHP中有着众多的二维码解码库&#xff0c;例如&#x…

前端将图片储存table表格中,页面回显

<el-table :data"tableData" v-loading"loading" style"width: 100%" height"calc(100vh - 270px)" :size"tableSize"row-dblclick"enterClick"><el-table-column prop"name" label"文档…

代码随想录Day30 贪心05 LeetCode T435无重叠区间 T763划分字母区间 T56 合并区间

LeetCode T435 无重叠区间 题目链接:435. 无重叠区间 - 力扣&#xff08;LeetCode&#xff09; 题目思路: 这题思路和昨天的打气球类似,我们需要按照左区间或者右区间进行排序,然后哦判断第i个区间的左端点和第i-1个区间的右端点的大小关系,,如果大于等于,那么就无需操作,一旦…

一文搞懂 MineCraft 服务器启动操作和常见问题 2023年10月

文章目录 前言1. 新建文件夹2. 创建 bat 文件3. 编辑 bat 文件4. 启动服务器5. 恭喜完成 文章持续更新中&#xff0c;如果你有问题可以通过 qq 1317699264 获取免费协助&#xff0c;解决的问题将会被更新到本文章中 前言 无论你是使用服务端整合包&#xff0c;还是从上一篇我的…

3.线性神经网络

#pic_center R 1 R_1 R1​ R 2 R^2 R2 目录 知识框架No.1 线性回归基础优化算法一、线性回归1、买房案例2、买房模型简化3、线性模型4、神经网络5、损失函数6、训练数据7、参数学习8、显示解9、总结 二、 基础优化算法1、梯度下降2、学习率3、小批量随机梯度下降4、批量大小5、…

Gerrit | 重磅! 2.x 版本升级到 3.x 版本----转

Gerrit | 重磅! 2.x 版本升级到 3.x 版本 为什么要做版本升级&#xff1f; 2.x known bugs 重大问题不一一列举&#xff0c;这里仅仅是举几个例子&#xff1a; 安全或权限问题&#xff1a;普通用户能看到敏感数据&#xff0c;例如看到其他用户的 hashed api 密码&#xff0c…