内存函数 memcpy、memmove 的简单模拟实现

news2025/1/11 17:59:15

一、memcpy 函数

数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。注意是以字节为单位进行拷贝。函数声明如下:

1、参数返回值解析

第二个参数 src:源地址,即你要从哪开始拷贝。

第三个参数 count:字节数,即从源地址起始位置开始一共要拷贝多少个字节。

第一个参数 dst:目的地址。也就是你拷贝的内容要放到哪。

返回值:返回目标空间的地址。

2、使用注意

该函数有如下两个注意事项:

  • 因为是内存函数,并不会以' \0 '作为结束符。
  • 不适合用于处理内存重叠的情况。举个例子,将源地址的第1 ~ 4个字节拷贝到源地址的第3 ~ 6的字节位置上,此时发生了内存重叠。(如果有内存重叠,则需要使用memmove函数)

3、模拟实现

因为是逐字节拷贝,所以解引用的时候,每次访问只能访问一个字节,前面在说指针类型的时候提到,指针类型决定了解引用每次可以访问多少个字节。

因此,在拷贝之前,先将源地址和目标地址的指针类型转化为 char* 类型,这样的话,每次解引用就只会访问一个字节。

#include <assert.h>
#include <stdio.h>

void* my_memcpy(void* dst, const void* src, size_t num)
{
	assert(dst && src);

	char* target = (char*)dst;		        // 转化为 char*,解引用时只会访问一个字节		
	const char* source = (const char*)src;  

	while (num--)
	{
		*target = *source;
		target++;
		source++;
	}
	return dst;
}

int main() {
	int dst[10] = { 0 };
	int src[5] = { 1,2,3,4,5 };

	my_memcpy(dst, src, sizeof(src));

	for (size_t i = 0; i < 10; i++)
	{
		printf("%d ", dst[i]);
	}
	return 0;
}

测试结果如下:

二、memmove 函数

1、认识memmove函数

当源地址和目标地址没有重叠的时候,memmove函数 和 memcpy函数的功能是一样的,但是如果遇到内存重叠的情况,比如将源地址的第1 ~ 4个字节拷贝到源地址的第3 ~ 6的字节位置上。

函数声明如下, 参数解析可以参考 memcpy 函数

 

2、memmove函数模拟实现

在模拟实现的时候,需要考虑的是,源地址和目的地址的位置。如果源地址在目的地址的左边,此时应该先将末尾的字节拷贝到目标地址的末尾。如果按正常顺序拷贝的话,就会变成右边这种情况,明明 3 也是要拷贝到目标地址的,但是现在却被新拷贝的覆盖了。

如果        边,应该是从后往前拷贝。这个时候相当于是把蓝色框里的内容向后移动了一样,最先移动的是最右边的内容。

 如果源地址在目的地址的右边,那么这个时候,就应该是从前往后拷贝。这个时候就相当于是把蓝色框里的内容向前移动了一样,最先移动的是最左边的内容。

void* my_memmove(void* dst, const void* src, size_t num)
{
	assert(dst && src);

	char* target = (char*)dst;
	const char* source = (const char*)src;


	if (src < dst)
	{
		// 源地址在目的地址的左边
		while (num--)
		{
			*(target + num) = *(source + num);
		}
	}
	else if (src > dst)
	{
		// 源地址在目的地址的右边
		while (num--)
		{
			*target = *source;
			target++;
			source++;
		}
	}
	else
	{
		// 无需拷贝,直接返回
		return (void*)src;
	}
}

int main() {
	int arr[8] = { 1,2,3,4,5,6,7,8 };

	my_memmove(arr, arr + 2 , 4 * sizeof(int));		//移动4个元素(16个字节)

	for (size_t i = 0; i < 8; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

 测试结果为:

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

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

相关文章

Qt 物联网系统界面开发 “ 2022湖南省大学生物联网应用创新设计竞赛技能赛 ——应用物联网的共享电动自行车 ”

文章目录前言一、实现效果二、程序设计1. 界面背景图设计2. 信号槽设计3. 定时器设计4. 动态曲/折线图的设计5. 摄像头扫码6. 注册设计7. 登录设计8. 巡检人员设计三、综合分析前言 本篇源于 “ 2022 湖南省大学生物联网应用创新设计竞赛技能赛参考样题 ” ——应用物联网的共享…

【git】git ssh 公钥私钥 在 windows和mac 双系统分别如何生成 以及对接各个平台说明

win和mac 双系统分别如何生成 git ssh 一、windows 生成 ssh 公钥私钥 windows版本需要下载git bash&#xff1a;https://gitforwindows.org/ 在 git bash 中输入如下指令&#xff1a; # 创建全局名称&#xff08;将会在你的git提交作者中显示&#xff09;git config --glo…

【allegro 17.4软件操作保姆级教程三】布局操作基础二

4精准定位与坐标定位 在设计中经常会有一些器件或结构孔要摆放在指定位置&#xff0c;如果用move命令用鼠标去移则很难定位完全&#xff0c;这时候就需要精准定位。 操作步骤为&#xff1a; 1、点击move命令&#xff0c;在option面板选择器件原点&#xff0c;这时器件就会悬停在…

策略分析中缺失值的处理方法

在日常的策略分析中&#xff0c;经常会碰到分析的变量出现缺失值的情况&#xff0c;如果对这些缺失值视而不见&#xff0c;则会对策略分析的结果造成一定的影响。那么我们如何处理缺失值呢&#xff1f;关注“金科应用研院”&#xff0c;回复“CSDN”领取“风控资料合集” 首先…

本地数据库IndexedDB - 学员管理系统之登录(一)

IndexedDB是浏览器提供的本地数据库&#xff0c;它可以被网页脚本创建和操作。IndexedDB允许存储大量数据&#xff0c;提供查找接口&#xff0c;还能建立索引。这些都是LocalStorage或Cookie不具备的。就数据库类型而言&#xff0c;IndexedDB不属于关系型数据库&#xff08;不支…

插入排序图解

七大排序之插入排序 文章目录七大排序之插入排序前言一、直接插入排序1.1 算法图解1.2 算法稳定性1.3 插入排序和选择排序相比到底优在哪&#xff1f;二、折半插入排序总结前言 博主个人社区&#xff1a;开发与算法学习社区 博主个人主页&#xff1a;Killing Vibe的博客 欢迎大…

springboot:实现文件上传下载实时进度条功能【附带源码】

0. 引言 记得刚入行的时候&#xff0c;做了一个文件上传的功能&#xff0c;因为上传时间较久&#xff0c;为了用户友好性&#xff0c;想要添加一个实时进度条&#xff0c;显示进度。奈何当时技术有限&#xff0c;查了许久也没用找到解决方案&#xff0c;最后不了了之。 近来偶…

全网最全面的pytest测试框架进阶-conftest文件重写采集和运行测试用例的hook函数

【文章末尾有.......】 使用pytest不仅仅局限于进行单元测试&#xff0c;作为底层模块可扩展性强&#xff0c;有必要理解其运行机制&#xff0c;便于进行二次开发扩展&#xff0c;通过文档的学习很容易理解。 构建一个简单的测试脚本 import pytest import requestsdef add(…

Hive数据倾斜常见场景及解决方案(超全!!!)

Hive数据倾斜常见问题和解决方案 文章目录 前言、一、Explain二、数据倾斜&#xff08;常见优化&#xff09;前言 Hive数据倾斜是面试中常问的问题&#xff0c;这里我们需要很熟练地能举出常见的数据倾斜的例子并且给出解决方案。 一、Explain 我们可以通过sql语句前面加expa…

公众号网课查题搭建方法

公众号网课查题搭建方法 本平台优点&#xff1a; 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a; 查题校园题库&#xff1a;查题校园题库后台&#xff08;…

QLC 闪存给主控带来了很大的难题?

前言 世界各大主流闪存厂商&#xff0c;如美光、海力士、铠侠和长江存储积极致力于QLC的研发&#xff0c;并相继推出了QLC SSD 产品。随着技术的不断进步&#xff0c;人们普遍担心的QLC擦写寿命少正逐渐被改善。QLC SSD 成本是最大的优势&#xff0c;不指望说替代 TLC SSD&…

408 | 【2011年】计算机统考真题 自用回顾知识点整理

选择题 T3&#xff1a;循环队列 不同指针指向&#xff0c;队列判空/判满条件 1. rear:指向队尾元素 front:指向队头元素前一个位置 &#xff08;1&#xff09;牺牲一个存储空间 &#xff08;2&#xff09;判空条件&#xff1a;front rear &#xff08;3&#xff0…

【RHCSA】管理Linux的联网

目录 rhel8与旧版本的区别 NetworkManager的特点 配置网络 (1)使用P命令配置临时生效的网络连接 (2)修改配置文件&#xff0c;前提是需要有network服务[不推荐] (3)nmcli(命令行工具) 网络测试命令 Ⅰ、使用ping命令测试网络的连通性 Ⅱ、使用tracepath命令跟踪并显示网…

2023最新SSM计算机毕业设计选题大全(附源码+LW)之java危险品运输车辆信息管理系统b2z1o

大学毕业设计&#xff0c;一般都是自己或者几个同学一起弄&#xff0c;lunwen都是去&#xff0c;百度&#xff0c;图书馆找很多资料参考&#xff0c;&#xff08;就是把里面都了&#xff0c;自己再按照各个意思重新表达&#xff09;&#xff0c;但是前提&#xff0c;提纲要想好…

【附源码】计算机毕业设计SSM微课程服务系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

神经网络芯片的单片机,什么是神经网络芯片

1、神经网络做图像分类一定要用到gpu吗&#xff1f; GPU最大的价值一直是“accelerating”(加速)&#xff0c;GPU不是取代CPU&#xff0c;而是利用GPU的并行计算架构&#xff0c;来将并行计算的负载放到GPU上来处理从而极大的提升处理速度。GPU本质上在异构计算架构上属于协处…

Hello Word你真的理解了么?今天教我的表弟,有些感悟

&#x1f36c;博主介绍 &#x1f468;‍&#x1f393; 博主主页&#xff1a;喵的主页 ✨主攻领域&#xff1a;【大数据】【java】【python】【面试分析】 Hello world1. 编写程序2. 打开命令行3. 运行 .class 文件4. 排查错误1. 编写程序 是不是都忘了我们初学时是打开记事本的…

9-1 Kubernetes二进制部署的Prometheus实现服务发现

文章目录前言创建用户复制Token配置文件全局配置Master节点发现Node节点发现Namespace Pod发现自定义Pod发现前言 在上一章节介绍了 8-5 在Prometheus实现Kubernetes-apiserver及Coredns服务发现 基于K8s集群内部安装的Prometheus&#xff0c;添加服务发现时更加方便。Prometh…

二叉树遍历原理 | 深度优先-广度优先 | 栈-队列

&#x1f497;wei_shuo的个人主页 &#x1f4ab;wei_shuo的学习社区 &#x1f310;Hello World &#xff01; 14天阅读挑战赛 文章目录二叉树遍历原理队列和栈区别深度优先遍历(DFS)前序遍历(根-左-右)中序遍历(左-根-右)后序遍历(左-右-根)广度优先遍历(BFS)逐层遍历(上-下 | …

软件研发人效提升之道,法,术,器,势

在华为的寒气传递之前&#xff0c;笔者已经在思考和实战如何提高研发人效。目的目标很明确: 提高软件研发人效&#xff0c;所谓的软件人效&#xff0c;简单粗暴的定义就是以最低单位人均成本&#xff0c;快速&#xff0c;高质量&#xff0c;高频率&#xff0c;安全地交付软件产…