【模拟实现C语言库函数】string.h中的内存操作函数

news2025/1/11 2:28:01

本篇文章目录

  • 相关文章
  • 1. 模拟 memcpy 内存拷贝
  • 2. 模拟 memmove 内存移动

相关文章

  1. 【C语言】数据在内存中是以什么顺序存储的?
  2. 【C语言】整数在内存中如何存储?又是如何进行计算使用的?
  3. 【C语言】利用void*进行泛型编程
  4. 【C语言】4.指针类型部分

使用内存库函数实际上要包含string.h头文件,这个大伙要注意。

1. 模拟 memcpy 内存拷贝

两个指针的指向必须是两块互相独立的内存区域,即两个不同的数组。
dest空间必须比src空间大;
bytes表示要从src拷贝到dest的字节数。

// void* 通用的泛型编程,可以接收任何指针
void* my_memcpy(void* dest, const void* src, size_t bytes) {
	assert(dest && src);
	if (dest == src) {
		return dest;
	}
	void* t = dest;
	while (bytes--) {
		// 不清楚void*接收的是什么类型指针,直接char*一个个字节拷贝。
		*(char*)dest = *(char*)src;
		dest = (char*)dest + 1;
		src = (char*)src + 1;
		/*
			((char*)dest)++;
			((char*)src)++;
			---------------
			++((char*)dest);
			++((char*)src);
			这两种写法换成c++都不行
		*/
	}
	return t;
}
int main() {
	int arr1[] = { 1, 2, 3, 4, 5, 6, 7, 8 };
	int arr2[] = { 1, 2, 3, 4};
	my_memcpy(arr1 + 4, arr2, sizeof(int) * 4);
	for (int i = 0; i < 8; i++) {
		printf("%d ", arr1[i]);
	}
	return 0;
}

在这里插入图片描述

对于标准的memcpy,如果不进行完善实际上是有问题的:假设有一个数组 int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8 },指针dest和指针src两个指针指向同一个数组内存,并且以字节为单位进行内存移动,会出现几种情况:

1.dest和src指向同一个地址,如my_memcpy(arr, arr, 4),等于啥也没动;

2.dest指向地址值大于src
 2.1 如my_memcpy(arr + 4, arr, 16),也就是将5 6 7 8改成1 2 3 4,结果为1 2 3 4 1 2 3 4;
 -----------------------------------------------------
 2.2 如my_memcpy(arr + 2, arr, 16),也就是将3 4 5 6改成1 2 3 4,即 1 2 1 2 3 4 7 8
 但结果会变成1 2 1 2 1 2 7 8,因为原来3 4位置被改成了1 2。

3.dest指向地址值小于src
 3.1 如my_memcpy(arr, arr + 4, 16),也就是将1 2 3 4改成5 6 7 8,结果为5 6 7 8 5 6 7 8;
 -----------------------------------------------------
 3.2 如my_memcpy(arr, arr + 2, 16),也就是将1 2 3 4改成3 4 5 6,结果为 3 4 5 6 5 6 7 8。

到这里会发现,也就只有2.2的结果不是我们想要的,这是因为自己实现的memcpy并不不完善,如果是string.h库函数中的memcpy则不存在这个问题。
在这里插入图片描述
本来是要将3 4 5 6改成1 2 3 4,结果改成了1 2 1 2。

事实上对于memcpy函数,C语言标准定义的是两个指针指向的内存位置不能是同一块区域,但显然对于vs2022的编译器而言是将memcpy完善了。但我们使用时还是尽量不要将两个指针指向同一个数组内的元素地址,毕竟要考虑到其它编译器并不一定完善。

画图分析2.2 my_memcpy(arr + 2, arr, sizeof(int) * 4):

在这里插入图片描述
图中每个格子代表arr数组中的一个元素,每个元素四个字节。我们利用这个简单的图分析上面模拟实现memcpy的代码,不难看出实际上拷贝是从前往后进行拷贝的,也就是从src、dest的起始位置开始往后拷贝。当拷贝完8个字节后,就变成了下面的样子:
在这里插入图片描述
这时的3和4早已被拷贝成了1和2,3和4不存在了。那么5和6就无法被拷贝成3和4了,自然而然也变成了1和2。

而解决这个问题,使用库函数memmove最好,对于这个函数,C语言的使用标准是这样的:两个指针既可以指向同一块内存区域,也可以像memcpy一样,两个指针指向不同内存区域。

2. 模拟 memmove 内存移动

如果自己实现memmove,通过上面例子出现的问题(同一块数组内存区域),如果要解决该问题,要考虑到的情况实际上也就是从前往后还是从后往前拷贝的问题,这个得由dest和src的地址大小比较后决定。

就对于上面模拟实现memcpy的问题,如果是从后往前拷贝,比如把6改成4,再把5改成3,再把3改成2,把2改成1,互不影响那么问题迎刃而解,但如果大伙认为真这么简单那就打错特错了。

对于内存而言以字节为单位,1个整型4个字节,我们实际上是要从最后一个字节开始往前一个个字节拷贝。arr数组的内存布局如下:
在这里插入图片描述
每一格都是1个字节,四格则构成一个完整的整型数据,也就是arr数组中的一个元素。格子中的值是用十六进制表示的,至于为什么是倒着存储的,这是因为当前机器以小端字节序存储数据(详细了解请看本篇文章最上面的 相关文章位置,第一个链接中的文章有解释)。

则对于dest地址值大于src的情况,可以这样拷贝:
在这里插入图片描述
而对于dest地址值小于src的情况,照常从前往后拷贝:
在这里插入图片描述

//模拟memmove(两个指针的指向可以是两块互相独立内存,也可以是同一块内存)
void* my_memmove(void* dest, const void* src, size_t bytes) {
	assert(dest && src);
	void* t = dest;
	// 从后面最后一个字节往前,将6改成4,再将5改成3,4改成2,3改成1,解决上面memcpy 2.2中的问题
	if (dest > src) {
		while (bytes--) { 
			*((char*)dest + bytes) = *((char*)src + bytes);
		}
	} 
	else if (dest < src) {     
		while (bytes--) {
			*(char*)dest = *(char*)src;
			dest = (char*)dest + 1;
			src = (char*)src + 1;
		}
	} 
	return t;
}

int main() {
	int arr[] = { 1, 2, 3, 4, 5, 6, 7, 8 };
	my_memmove(arr, arr + 2, sizeof(int) * 4);
	for (int i = 0; i < 8; i++) {
		printf("%d ", arr[i]);
	}
	return 0;
}

在这里插入图片描述
成功把3 4 5 6改成1 2 3 4!


在这里插入图片描述
把1 2 3 4改成3 4 5 6也没问题!

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

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

相关文章

postgresql pgsql 连接池 pgBouncer(详细)

适用连接池可以有效的降低反复连接造成的损耗 2023.9.28更新&#xff0c;演示&#xff1a;1.20.1版本 没有安装pgsql的可以参考&#xff1a;pgsql编译安装 一、编译安装 包安装更简单&#xff0c;就是不能选版本 1.pgBouncer下载地址 链接&#xff1a;github项目地址 链接&a…

2023软件测试面试八股文大全(含答案)

1、B/S架构和C/S架构区别 B/S 只需要有操作系统和浏览器就行&#xff0c;可以实现跨平台&#xff0c;客户端零维护&#xff0c;维护成本低&#xff0c;但是个性化能力低&#xff0c;响应速度较慢 C/S响应速度快&#xff0c;安全性强&#xff0c;一般应用于局域网中&#xff0c…

java一体化智慧工地信息管理平台源码 智慧工地APP源码

智慧工地云平台是专为建筑施工领域所打造的一体化信息管理平台。通过大数据、云计算、人工智能、物联网和移动互联网等高科技技术手段&#xff0c;将施工区域各系统数据汇总&#xff0c;建立可视化数字工地。同时&#xff0c;围绕人、机、料、法、环等各方面关键因素&#xff0…

探索ClickHouse——连接Kafka和Clickhouse

安装Kafka 新增用户 sudo adduser kafka sudo adduser kafka sudo su -l kafka安装JDK sudo apt-get install openjdk-8-jre下载解压kafka 可以从https://downloads.apache.org/kafka/下找到希望安装的版本。需要注意的是&#xff0c;不要下载路径包含src的包&#xff0c;否…

QQ聊天记录文件怎么恢复?这3个方法亲测有效

QQ为用户提供了聊天、语音、视频、在线游戏、社交分享等丰富的功能&#xff0c;满足了用户的各种通讯以及娱乐需求。无论是现在还是过去&#xff0c;QQ仍然在我们的生活中扮演着重要的角色。 如果在使用QQ的过程中&#xff0c;发现文件过期或者被删除了该怎么办&#xff1f;qq…

480439-15-4,一种具有荧光单体的pH敏感性染料Fluorescein O-methacrylate

产品简介&#xff1a;荧光素O-甲基丙烯酸酯是一种具有荧光单体的pH敏感性染料。它可以通过490 nm的激发光谱和520 nm的发射光谱进行表征。它具有荧光素&#xff0c;其是一种负电荷最少的指示剂。它的特性包括生物相容性、无毒性&#xff0c;以及在水溶液中的良好分散性。 CAS号…

Vue3最佳实践 第六章 Pinia,Vuex与axios,VueUse 1(Pinia)

Pinia状态管理 在 Vue3 中项目中组件之间传递数据时&#xff0c;可以使用 Props 和 Emit&#xff0c;还可以使用 Provide/Inject 来代替 Props 和 Emit。Props 和 Emit 只能在具有父子关系的组件之间传递数据&#xff0c;所以层级越深&#xff0c;过程就越复杂。为了解决此类问…

brew 安装MySQL 5.7

写在前面&#xff1a;博主是一只经过实战开发历练后投身培训事业的“小山猪”&#xff0c;昵称取自动画片《狮子王》中的“彭彭”&#xff0c;总是以乐观、积极的心态对待周边的事物。本人的技术路线从Java全栈工程师一路奔向大数据开发、数据挖掘领域&#xff0c;如今终有小成…

【云原生】k8s集群调度

目录 一、调度约束 1.1List-Watch工作机制 1.2调度过程 二、指定调度节点 2.1修改成 nodeSelector 调度方式 三、亲和性 &#xff08;1&#xff09;节点亲和性 &#xff08;2&#xff09;Pod 亲和性 3.1 键值运算关系 四、污点(Taint) 和 容忍(Tolerations) 4.1污点(…

搭建 Prometheus 对服务进行监控

前言&#xff1a; 服务平时没啥问题&#xff0c;一到过节我放假&#xff0c;它也想放假&#xff0c;所以只能找个监工看着了。当前市面上主流的监工方案是 zabbix 和 prometheus&#xff0c;没有优劣之分&#xff0c;只是适用场景略有区别。我这边的需求就主要是监控服务的端口…

温度与振动监测技术在电机故障智能诊断中的应用

在现代工业生产中&#xff0c;电机是许多设备和机械的核心驱动力。然而&#xff0c;电机故障可能会导致生产中断、设备损坏以及生产成本的增加。为了避免这些问题&#xff0c;工业设备状态监测技术应运而生。本文将探讨如何利用先进的设备状态监测技术&#xff0c;尤其是温度和…

各种业务场景调用API代理的API接口教程

API代理的API接口在各种业务场景中具有广泛的应用&#xff0c;本文将介绍哪些业务场景可以使用API代理的API接口&#xff0c;并提供详细的调用教程和代码演示&#xff0c;同时&#xff0c;我们还将讨论在不同场景下使用API代理的API接口所带来的好处。 哪些业务场景可以使用AP…

C# 字符串和正则表达式

C# 字符串和正则表达式 System.String 类StringBuilder 成员格式化字符串正则表达式 System.String 类 StringBuilder 成员 格式化字符串 正则表达式

央国企信创改造难在何处?先行建设国产身份域管可少走弯路

据统计&#xff0c;全球超 91% 的具规模企业将 Microsoft Active Directory &#xff08;微软AD&#xff09;作为数字化身份的基础底座。通常企业达到 300 人以上规模开始建设 AD&#xff0c;而高科技企业早在 50-60 人左右规模时就开始搭建。 AD身份域管是企业身份的事实标准 …

货物寄到英国选择什么物流比较划算?

随着全球化的发展&#xff0c;越来越多的企业开始将产品销售到海外市场&#xff0c;其中英国作为一个重要的贸易伙伴&#xff0c;吸引了大量的中国企业的关注。然而&#xff0c;如何将货物安全、快速地运送到英国&#xff0c;成为了众多企业面临的一个问题。那么&#xff0c;货…

拥抱数字正义时代,看AIGC如何驱动法律变革

人工智能&#xff0c;作为科技领域的代表&#xff0c;目前正在逐步渗透并应用于各个领域。大到政府的社会治理&#xff0c;小到提问的识别延伸&#xff0c;AI已悄然走进了生活中的各个领域。单说人民法院的司法工作中&#xff0c;随机分案的程序设计、音字转换的功能实现、裁判…

04、EL和JSTL核心技术

目录 1 EL表达式&#xff08;熟悉&#xff09; 1.1 基本概念 1.2 主要功能 1.3 访问内置对象的数据 1.3.1访问方式 1.3.2 执行流程 1.4 访问请求参数的数据 1.5 访问Bean对象的属性 1.5.1 访问方式 1.5.2 主要区别 1.6 访问集合中的数据 1.7 常用的内置对象 …

二叉树题目:二叉树剪枝

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;二叉树剪枝 出处&#xff1a;814. 二叉树剪枝 难度 4 级 题目描述 要求 给定二叉树的根结点 root \texttt{root} root&#xff0c;返回移除了所有…

LEO天线,全球市场总体规模,头部前八大厂商排名及市场份额

LEO天线全球市场总体规模 据QYResearch调研团队最新报告“全球LEO天线市场报告2023-2029”显示&#xff0c;预计2029年全球LEO天线市场规模将达到3545.3百万美元&#xff0c;未来几年年复合增长率CAGR为29.6%。 主要驱动因素: 近年来&#xff0c;全球航天工业的投资激增&#…

React antd Table点击下一页后selectedRows丢失之前页选择内容的问题

一、问题 使用了React antd 的<Table>标签&#xff0c;是这样记录选中的行id与行内容的&#xff1a; <TabledataSource{data.list}rowSelection{{selectedRowKeys: selectedIdsInSearchTab,onChange: this.onSelectChange,}} // 表格是否可复选&#xff0c;加 type: …