内存函数memcpy和memmove的讲解

2024-03-18 04:44

本文主要是介绍内存函数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就代表来到了这个要拷贝目标的最后一个字节,和要拷贝对象的最后一个字节,这样来实现倒着拷贝。
以上就是对这两个函数的讲解了,如果有什么不足的欢迎指正。

这篇关于内存函数memcpy和memmove的讲解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/821215

相关文章

NameNode内存生产配置

Hadoop2.x 系列,配置 NameNode 内存 NameNode 内存默认 2000m ,如果服务器内存 4G , NameNode 内存可以配置 3g 。在 hadoop-env.sh 文件中配置如下。 HADOOP_NAMENODE_OPTS=-Xmx3072m Hadoop3.x 系列,配置 Nam

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

OpenCV结构分析与形状描述符(11)椭圆拟合函数fitEllipse()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C++11 算法描述 围绕一组2D点拟合一个椭圆。 该函数计算出一个椭圆,该椭圆在最小二乘意义上最好地拟合一组2D点。它返回一个内切椭圆的旋转矩形。使用了由[90]描述的第一个算法。开发者应该注意,由于数据点靠近包含的 Mat 元素的边界,返回的椭圆/旋转矩形数据

JVM内存调优原则及几种JVM内存调优方法

JVM内存调优原则及几种JVM内存调优方法 1、堆大小设置。 2、回收器选择。   1、在对JVM内存调优的时候不能只看操作系统级别Java进程所占用的内存,这个数值不能准确的反应堆内存的真实占用情况,因为GC过后这个值是不会变化的,因此内存调优的时候要更多地使用JDK提供的内存查看工具,比如JConsole和Java VisualVM。   2、对JVM内存的系统级的调优主要的目的是减少

JVM 常见异常及内存诊断

栈内存溢出 栈内存大小设置:-Xss size 默认除了window以外的所有操作系统默认情况大小为 1MB,window 的默认大小依赖于虚拟机内存。 栈帧过多导致栈内存溢出 下述示例代码,由于递归深度没有限制且没有设置出口,每次方法的调用都会产生一个栈帧导致了创建的栈帧过多,而导致内存溢出(StackOverflowError)。 示例代码: 运行结果: 栈帧过大导致栈内存

理解java虚拟机内存收集

学习《深入理解Java虚拟机》时个人的理解笔记 1、为什么要去了解垃圾收集和内存回收技术? 当需要排查各种内存溢出、内存泄漏问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就必须对这些“自动化”的技术实施必要的监控和调节。 2、“哲学三问”内存收集 what?when?how? 那些内存需要回收?什么时候回收?如何回收? 这是一个整体的问题,确定了什么状态的内存可以