【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】

本文主要是介绍【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

欢迎来CILMY23的博客喔,本篇为​【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】,图文讲解四种内存函数,带大家更深刻理解C语言中内存函数的操作,感谢观看,支持的可以给个一键三连,点赞关注+收藏。 

前言

在结束上一期字符系列篇后,本篇我们将了解四种内存操作的函数,它们分别是memcpy,memmove,memset和memcmp。

目录

一、memcpy

memcpy的介绍和使用 

 memcpy的模拟实现

二、memmove

 memmove的介绍和使用

 memmove的模拟实现

三、memset

四、memcmp 


一、memcpy

 memcpy可以在cplusplus网站查询,memcpy - C++ Reference (cplusplus.com)

函数原型如下:

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

函数介绍如下:

 

函数返回值和使用案例如下:

 

memcpy的介绍和使用 

 memcpy是一个复制内存空间的函数,Copy block of memory,复制内存块,将 num 字节的值从指向的位置直接复制到目标指向的内存块。

那具体是怎么使用的呢?

我们来看一个整型数组的使用案例:

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

结果如下:

 memcpy的模拟实现

思路: 因为我们并不知道要接收什么样的数据类型,所以可以用void *来定义数据类型,因为void*不能直接加减整数,所以我们要将其转换成char * 的一个字节指针变量,然后进行加减,最后赋值给自己。

void* my_memcpy(void* dest, const void* src, size_t num)
{assert(dest && src);void* ret = dest;while (num--){*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}return ret;
}

 写完后,我们看案例,如果目标空间和源空间重叠了呢?

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

结果如下:

我们用自己的拷贝呢?

int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };my_memcpy(arr1+3, arr1, 20);for (int i = 0; i < 10; i++){printf("%d ", arr1[i]);}return 0;
}

结果如下: 

 

我们来看过程图:

总结:

1.memcpy的使用需要包括头文件string.h

2.函数memcpy从source的位置开始向后复制num个字节的数据到destination指向的内存位置。
3.memcpy函数在遇到  '\0' 的时候并不会停下来。
4.如果source和destination有任何的重叠,复制的结果都是未定义的。

5.memcpy的返回值是目标空间的起始地址

二、memmove

为了解决上述情况,我们需要用到memmove来解决内存重叠的问题,memmove可以在cplusplus网站查询,memmove - C++ Reference (cplusplus.com)

函数原型如下:

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

函数介绍如下:

函数返回值和使用案例如下:

 memmove的介绍和使用

 memmove可以解决出现内存重叠空间的情况,将 num 字节的值从指向的位置复制到目标指向的内存块。复制就像使用中间缓冲区一样进行,从而允许目标重叠。

int main()
{int arr1[] = { 1,2,3,4,5,6,7,8,9,10 };memmove(arr1+3, arr1, 20);for (int i = 0; i < 10; i++){printf("%d ", arr1[i]);}return 0;
}

结果如下: 

 memmove的模拟实现

 思路:为了解决重叠问题,我们得对src和dest的目标空间起始位置进行讨论,分两种情况,从后往前或者从前往后复制

我们发现当dest在src的右边就需要从后往前拷贝 

 

而dest在src的左边就需要从前往后拷贝 

 

当dest和src不重叠的时候,无论是从前往后,还是从后往前都可以。 

 所以一共有两种方案

方案一,我们采取dest在src前面的情况,然后其余只采用从后往前

方案二、我们采取dest >= src,并且,dest <= (char*)src+ num,

//方案一
void* my_memmove(void* dest, const void* src, size_t num)
{assert(dest && src);void* ret = dest;if (dest < src){//从前向后while (num--){*(char*)dest = *(char*)src;dest = (char*)dest + 1;src = (char*)src + 1;}}else{//从后向前while (num--){*((char*)dest + num) = *((char*)src + num);}}return ret;
}
//方案二
void* my_memmove(void* dest, const void* src, size_t num)
{assert(dest && src);void* ret = dest;if (dest > src && dest <= (char*)src +num){//从后向前while (num--){*((char*)dest + num) = *((char*)src + num);}}else{//从前向后while (num--){*(char*)dest = *(char*)src;			dest = (char*)dest + 1;src = (char*)src + 1;}}return ret;
}

二者最后的结果如下所示:

总结:
1.如果源空间和⽬标空间出现重叠,就得使用memmove函数处理。

2.memmove的使用需要包括头文件string.h

3.memmove将 num 字节的值将源指向的位置复制到目标指向的内存块。复制就像使用中间缓冲区一样进行,从而允许目标和源空间重叠。

4.memmove和memcpy函数一样在遇到  '\0' 的时候并不会停下来。

5.memmove的返回值是目标空间的起始地址

三、memset

 memset可以在cplusplus网站查询,memset - C++ 参考 (cplusplus.com)

函数原型如下:

void * memset ( void * ptr, int value, size_t num );

函数介绍如下:

 

函数返回值和使用案例如下:

 

函数的使用和介绍 

memset是用来填充内存的,填充内存的值就是函数参数中的value

#include<stdio.h>
#include<string.h>int main()
{char arr[] = "hello CILMY23";memset(arr, '1', 5);for (int i = 0; i < 13; i++){printf("%c ", arr[i]);}return 0;
}

 结果如下:

总结:

1.memset是用来设置内存的,将内存中的值以字节为单位设置成想要的内容。

2.memset的使用需要包括头文件string.h

3.memset的返回是原空间的地址

4.memset只能对字节进行操作,操作的是一个字节数

四、memcmp 

 memcmp可以在cplusplus网站查询,memcpy - C++ Reference (cplusplus.com)

函数原型如下:

int memcmp ( const void * ptr1, const void * ptr2, size_t num );

函数介绍如下:

函数返回值和使用案例如下:

 memcmp的使用

#include<stdio.h>
#include<string.h>int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 1,5,6,7,8, };int ret = memcmp(arr1, arr2,5);printf("%d ", ret);return 0;
}

总结:

1.memcmp是比较从ptr1和ptr2指针指向的位置开始,向后的num个字节

2.memcmp的使用需要包括头文件string.h

3.memcmp的返回值是ptr1大于ptr2返回大于0的值,小于返回小于0的值,相等返回0

4.memcmp比较的是字节内容

感谢各位同伴的支持,本期内存函数篇就讲解到这啦,如果你觉得写的不错的话,可以给个一键三连,点赞关注+收藏,若有不足,欢迎各位在评论区讨论。   

这篇关于【C语言】内存操作,内存函数篇---memcpy,memmove,memset和memcmp内存函数的使用和模拟实现【图文详解】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux系统性能检测命令详解

《Linux系统性能检测命令详解》本文介绍了Linux系统常用的监控命令(如top、vmstat、iostat、htop等)及其参数功能,涵盖进程状态、内存使用、磁盘I/O、系统负载等多维度资源监控,... 目录toppsuptimevmstatIOStatiotopslabtophtopdstatnmon

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

Java中的数组与集合基本用法详解

《Java中的数组与集合基本用法详解》本文介绍了Java数组和集合框架的基础知识,数组部分涵盖了一维、二维及多维数组的声明、初始化、访问与遍历方法,以及Arrays类的常用操作,对Java数组与集合相... 目录一、Java数组基础1.1 数组结构概述1.2 一维数组1.2.1 声明与初始化1.2.2 访问

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

Ubuntu如何分配​​未使用的空间

《Ubuntu如何分配​​未使用的空间》Ubuntu磁盘空间不足,实际未分配空间8.2G因LVM卷组名称格式差异(双破折号误写)导致无法扩展,确认正确卷组名后,使用lvextend和resize2fs... 目录1:原因2:操作3:报错5:解决问题:确认卷组名称​6:再次操作7:验证扩展是否成功8:问题已解

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构