【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)

函数原型如下:

cpp 复制代码
void * memcpy ( void * destination, const void * source, size_t num );

函数介绍如下:

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

memcpy的介绍和使用

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

那具体是怎么使用的呢?

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

cpp 复制代码
#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 * 的一个字节指针变量,然后进行加减,最后赋值给自己。

cpp 复制代码
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;
}

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

cpp 复制代码
#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;
}

结果如下:

我们用自己的拷贝呢?

cpp 复制代码
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)

函数原型如下:

cpp 复制代码
void * memmove ( void * destination, const void * source, size_t num );

函数介绍如下:

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

memmove的介绍和使用

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

cpp 复制代码
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,

cpp 复制代码
//方案一
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;
}
cpp 复制代码
//方案二
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)

函数原型如下:

cpp 复制代码
void * memset ( void * ptr, int value, size_t num );

函数介绍如下:

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

函数的使用和介绍

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

cpp 复制代码
#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)

函数原型如下:

cpp 复制代码
int memcmp ( const void * ptr1, const void * ptr2, size_t num );

函数介绍如下:

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

memcmp的使用

cpp 复制代码
#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比较的是字节内容
感谢各位同伴的支持,本期内存函数篇就讲解到这啦,如果你觉得写的不错的话,可以给个一键三连,点赞关注+收藏,若有不足,欢迎各位在评论区讨论。

相关推荐
weixin_472339465 小时前
高效处理大体积Excel文件的Java技术方案解析
java·开发语言·excel
枯萎穿心攻击6 小时前
响应式编程入门教程第二节:构建 ObservableProperty<T> — 封装 ReactiveProperty 的高级用法
开发语言·unity·c#·游戏引擎
Eiceblue7 小时前
【免费.NET方案】CSV到PDF与DataTable的快速转换
开发语言·pdf·c#·.net
m0_555762908 小时前
Matlab 频谱分析 (Spectral Analysis)
开发语言·matlab
学不动CV了8 小时前
ARM单片机启动流程(二)(详细解析)
c语言·arm开发·stm32·单片机·51单片机
浪裡遊9 小时前
React Hooks全面解析:从基础到高级的实用指南
开发语言·前端·javascript·react.js·node.js·ecmascript·php
lzb_kkk9 小时前
【C++】C++四种类型转换操作符详解
开发语言·c++·windows·1024程序员节
好开心啊没烦恼10 小时前
Python 数据分析:numpy,说人话,说说数组维度。听故事学知识点怎么这么容易?
开发语言·人工智能·python·数据挖掘·数据分析·numpy
猫猫的小茶馆10 小时前
【STM32】通用定时器基本原理
c语言·stm32·单片机·嵌入式硬件·mcu·51单片机
简佐义的博客10 小时前
破解非模式物种GO/KEGG注释难题
开发语言·数据库·后端·oracle·golang