当前位置: 首页 > article >正文

内存函数memcpy和memmove的讲解

什么是memcpy函数

在cplusplus官网上是这样介绍的
在这里插入图片描述
这里的意思是memcpy会从开始位置复制若干个字节到终止的内存位置,这个函数在遇到\0时不会停下来,如果在复制的时候出现内存重叠的时候结果都是未定义的,也就是说我们有一个数组arr里面放有1,2,3,4,5,6个元素,现在我们把他复制20个字节到arr2数组里面,arr2里面就存放了1,2,3,4,5个元素,因为这个是按字节复制的一个整形占4个字节那么就是5个元素,如果把arr数组从首地址开始,复制到arr+2那个空间里面,这里就发生了内存重叠,结果就是未定义的。我们来举一个例子,来看看具体怎么使用。

# include<string.h>
# include<stdio.h>
int main()
{
	int arr[] = { 1,2,3,4,5 };
	int arr2[10] = { 0 };
	int sz = 20;
	memcpy(arr1, arr, 20);
	for (int i = 0; i < 10; i++)
	{
		printf("%d", arr2[i]);
	}
	return 0;
}

这个函数包含的头文件是string.h,我们看看运行结果。
在这里插入图片描述
这里我们发现我们成功复制了arr数组里面的20个字节到arr2里面并成功打印出了1,2,3,4,5这个元素,我们这里数组给的空间比较大,后面空余的空间会自动补0。
接下来我们看看memcpy函数的模拟实现

memcpy函数的模拟实现

我们来看看这张图
在这里插入图片描述
我们看到这个函数需要传的参数都需要转换成void的指针,这是因为我们不仅仅只能拷贝整形数据,还要能拷贝字符型数据等,所以我们要用void指针,下面来看看代码

#include <stdio.h>
#include <assert.h>
void* my_memcpy(void* start, void* go_to, size_t num)
{
	void* ret = go_to;
	assert(start != NULL && go_to != NULL);
    for(num ; num > 0 ;num--)
	{
		*(char*)go_to = *(char*)start;
		go_to = (char*)go_to + 1;
		start = (char*)start + 1;
	}
	return ret;
}
int main()
{
	int arr1[] = {1,2,3,4,5,6,7,8,9,10};
	int arr2[10] = { 0 };
	int num = 20;
	my_memcpy(arr1, arr2,num);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr2[i]);
	}
	return 0;
}

这里我们用start来代替官网中使用的scr,goto代替destination

void* my_memcpy(void* start, void* go_to, size_t num)

这里的参数就对应上面说的要变成void指针
首先把go_to的首地址存起来,我们最终是要返回go_to的,然后为了保证指针的有效性我们用assert来断言一下,防止指针为空然后下面就进入到拷贝环节了,我们用了一个for循环我们把他强制转换为char
这是因为char*类型的指针解引用他只向后偏移一个字节,这样我们可以拿这一个字节做一个标准一个单位来完成我们整形,字符型等元素的拷贝。最后就是打印数组了。下面是memmove函数讲解,这个的模拟实现要复杂一些。

什么是memmove函数

我们可以看到在c++官网是这样定义的
在这里插入图片描述
这个就可以用来在有重叠的空间进行拷贝内容,我们来看看是怎么用的他包含的头文件还是string.h

# include<string.h>
# include<stdio.h>
int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int sz = 20;
	memmove(arr+2, arr, sz);
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr[i]);
	}
	return 0;
}

运行结果
在这里插入图片描述
这里我们在同一内存空间成功拷贝了,接下来看看模拟实现

# include<stdio.h>
# include<assert.h>
my_memove(void* start, void* go_to, size_t num)
{
	void* ret = go_to;
	assert(start != NULL && go_to != NULL);
	if(start>go_to)
	{
		while(num-- != 0)
		{
			*(char*)go_to = *(char*)start;
			start = (char*)start + 1;
			go_to = (char*)go_to + 1;
		}
	}
	else
	{
		while (num--)
		{
			*((char*)go_to + num) = *((char*)start + num);
		}
	}
	return ret;
}
int main()
{
	int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };
	my_memove(arr1, arr1+2, 5*sizeof(int));
	for (int i = 0; i < 10; i++)
	{
		printf("%d ", arr1[i]);
	}
	return 0;
}

运行结果
在这里插入图片描述
这里我们需要知道是怎么拷贝的,我们需要分两种情况首先我们是从前往后拷贝,还是从后往前拷贝,通过分析在go_to小于start的时候就从后往前拷贝,当go_to大于start的时候从前往后拷贝这就是这个模拟实现最重要的地方,其他的就好理解了


*((char*)go_to + num) = *((char*)start + num);

这段代码的意思就是从后往前拷贝这里加上一个num就代表来到了这个要拷贝目标的最后一个字节,和要拷贝对象的最后一个字节,这样来实现倒着拷贝。
以上就是对这两个函数的讲解了,如果有什么不足的欢迎指正。


http://www.kler.cn/a/272004.html

相关文章:

  • 新浪安卓(Android)开发面试题及参考答案(68道题,9道手撕题)
  • 实战演示:利用ChatGPT高效撰写论文
  • 从零安装 LLaMA-Factory 微调 Qwen 大模型成功及所有的坑
  • WPF 引发类型为“System.Windows.Forms.AxHost+InvalidActiveXStateException”的异常 解决办法
  • 软件测试入门—用例设计中的场景图和状态迁移图
  • 将UI界面交给第三方库
  • 科技回顾,飞凌嵌入式受邀亮相第八届瑞芯微开发者大会「RKDC2024」
  • scrcpy远程投屏控制Android
  • 计算机等级考试:信息安全技术 知识点十二
  • vue2+vant2+Laravel7 实现多图上传到七牛云
  • 教你申请腾讯云免费服务器,准备好账号
  • SpringBoot(整合MyBatis + MyBatis-Plus + MyBatisX插件使用)
  • echo,date,bc命令详解
  • 模型、算法、数据模型、模型结构是什么?它们之间有什么关联和区别?
  • git |常用命令
  • 【Python】新手入门学习:什么是相对路径,应用相对路径有哪些注意事项
  • 2024年3月职业健康安全管理体系基础考试真题
  • Golang 中 map[string]string 如何在 TOML 文件中配置
  • cannot find -xml2: No such file or directory的解决方法
  • redis的基本知识点
  • node.js快速入门-day03
  • Python零基础---爬虫技术相关
  • LangChain支持Claude3接口
  • 线程常用方法
  • CentOS7.9 安装SIPp3.6
  • 力扣细节题:字符串中的最大奇数