C语言进阶第六篇【内存函数】

简介: C语言进阶第六篇【内存函数】

1. 内存拷贝库函数memcpy


❤️void * memcpy ( void * dest, const void * src, size_t num )

⭐️函数memcpy从src的位置开始向后复制num个字节的数据到dest的内存位置。

⭐️这个函数在遇到 '\0' 的时候并不会停下来。

⭐️如果src和dest有任何的重叠,复制的结果都是未定义的。


1.1 memcpy库函数的一般使用


对于memcpy库函数的使用我们可以对比着strcpy库函数的学习;都是拷贝库函数:


(1)strcpy库函数是字符串拷贝,只能拷贝字符串。


(2)memcpy库函数是内存拷贝库函数;可以拷贝任意的类型,适用范围更广;这里拿整型数据举例。


1.1.1 memcpy拷贝整型数据

f2ca5abd3c4048a6a2334823e161ae33.png


这里我们一定要明确,操作的是以字节为单位的!我们要拷贝前5个整型数据,实际上就是拷贝前20个字节过去!


1.1.2 memcpy拷贝字符型数据


69a7f7fad02949e7ad13b5413610ddb8.png


对于字符型数据,一个字符就是占一个字节;所以我们当我们要拷贝前5个字符,实际上就是拷贝5个字节过去!


1.2 memcpy库函数的模拟实现

既然要模拟实现,我们还是先看一下,库函数里的memcpy的定义:void * memcpy ( void * dest, const void * source, size_t num );因为我们要拷贝各种各样的数据,所以对于类型我们就写成void*,在使用时直接强制类型转化就行啦!


19ddf7d646ed4b5f8df48077a8f613bf.png


1、因为我们并不知道要拷贝什么样的数据,所以我们在设计参数的时候,就设计void*类型;像一个垃圾桶一样,什么类型都能接收;


2、接收以后在统一强制类型转换为char*类型;一个字节一个字节进行操作;


3、既然是void*返回类型,不是void,所以我们必须返回一个地址,不妨就返回目标空间的起始地址;


4、这里我使用了三种方法,觉得那种方法更好理解,就使用哪种方法!


1.3 利用memcpy库函数实现自己拷贝自己

既然strcpy能够利用strncpy来实现自己对自己的拷贝,那么memcpy可以实现自己对自己的拷贝吗?我们不妨来试验一下!


1.3.1 利用库函数来自己拷贝自己


81ec5ffd59204799acd88346e3c0363f.png


memcpy虽然可以实现自己对自己的拷贝,但是memcpy库函数一般用在拷贝不重叠的内存;不用来自己拷贝自己!把这种重叠的内存拷贝交给另一个库函数memmove,它们是有分工的!


1.3.2 利用模拟的库函数来自己拷贝自己


14fa2261df894d9bafab574a3131a342.png

我们发现利用自己模拟的memcpy库函数,打印的结果却不是我们想要的,为什么呢?因为我们在拷贝时1 2把3 4数据就覆盖掉了,所以后面的拷贝也就是1 2来回重复,直到要拷贝的个数结束 !那么如何模拟实现自己拷贝自己的功能呢?我们会在下面memmove的模拟实现讲到!


2. 内存拷贝库函数memmove

❤️void * memmove ( void * dest, const void * src, size_t num )

⭐️和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。

⭐️如果源空间和目标空间出现重叠,一般使用memmove函数处理。


2.1 memmove库函数的一般使用

memmove库函数的使用和memcpy的使用很相似;主要的区别就是:一般对于拷贝内存重叠的数据,我们使用memmove来实现;当然memcpy也是可以实现的!


aa7e7ffadd194346b1d1eaec45fc25bb.png


2.2 memmove库函数的模拟实现

1、如果要拷贝的数据有交叉重叠:


664ae3b4dcae4f4986d82d023baf794b.png


2、如果要拷贝的数据没有交叉重叠

b822176e807347489e7c1641c8861710.png



我们不妨总结一下上面两种情况:

bdad47adf6204273b701ba8d10bf140f.png



具体代码:


dba0e450d2dd4a21b8b8a3dcf5fca9e3.png


3. 内存比较库函数memcmp

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


⭐️比较从ptr1和ptr2指针开始的num个字节


3.1 memcmp的一般使用

我们还是和字符串比较函数strcmp函数对比着学习,strcmp是字符串比较函数,返回<0    ==0        >0三种值!memcmp用法也是一样的,不同的是,memcmp可以比较任何的数据类型;返回值如下:


36e1a8aebed34858b96fde7e94200b69.png


3.1.1 比较整型数据

18652637b236468796f5c1cebf421530.png


3.1.2 比较字符串型数据

838b29b43ea5442f8b5adda614397d88.png


看出区别没?对于不同的数据,我们同样是比较前4个字节,不同的数据类型得到的结果是不一样的!


(1)对于整型数据我们比较4个字节,其实就是比较一个int整型;


(2)对于字符型数据我们比较4个字节,其实就是比较四个字符型!


4. 内存设置库函数memset

❤️void *memset( void *dest, int c, size_t count )


⭐️对于内存设置库函数memset最多的应用就是用做初始化!


4.1 memset的一般使用

4.1.1 设置整型数据

比如一个整型数据,我们以前都是利用for循环来进行初始化;怎么利用memse来进行初始化呢?我们拿两个例子对比一下:


1、利用for循环来初始化


091dcb4401924498952d0dfe0f10a699.png


2、 利用memset库函数进行初始化

aabe42264a2440f5a6cd13ea94880869.png


4.1.2 设置字符型数据

7f0bfe1bb8a74c29bc0e498bbf26802b.png



总结:

这一期我们讲解了内存函数:memcpy拷贝库函数、memmove拷贝库函数、memcmp比较库函数、memset内存设置库函数;我们主要掌握他们的应用和前两个的模拟实现!希望对大家有所帮助!


相关文章
|
10天前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
24 3
|
1天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
17 10
|
1天前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
4天前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。
|
11天前
|
编译器 程序员 C语言
深入C语言:动态内存管理魔法
深入C语言:动态内存管理魔法
|
6天前
|
C语言
保姆级教学 - C语言 之 动态内存管理
保姆级教学 - C语言 之 动态内存管理
11 0
|
10天前
|
C语言
C语言函数
C语言函数
10 0
|
11天前
|
存储 C语言
深入C语言内存:数据在内存中的存储
深入C语言内存:数据在内存中的存储
|
11天前
|
C语言 C++
c语言回顾-内存操作函数
c语言回顾-内存操作函数
34 0
|
11天前
|
C语言
c语言回顾-函数递归(下)
c语言回顾-函数递归(下)
29 0