作者:喜欢猫咪的的程序员
专栏:《C语言》
喜欢的话:世间因为少年的挺身而出,而更加瑰丽。? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ——《人民日报》

前言:什么是动态内存?
动态内存是指在堆上分配的内存,而静态内存是指在栈上分配的内存。 前面所写的程序大多数都是在栈上分配的,比如局部变量、形参、函数调用等。 栈上分配的内存是由系统分配和释放的,空间有限,在复合语句或函数运行结束后就会被系统自动释放。
动态内存分配的函数:
malloc函数、free函数、calloc函数、realloc函数。
这4个函数的头文件都是#include<stdlib.h>
malloc函数:

这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
- 这个连续的空间大小就是size个字节,所以malloc函数是开辟size个字节的一个函数。
malloc函数如何使用?

?如上图:我们开辟了40个字节的动态空间。
malloc函数使用时,可能会出现的问题:
malloc这个函数的返回类型是void*,但void*类型我们不能进行运用。
因为这个函数向内存申请一块连续可用的空间:
- 如果开辟成功,则返回一个指向开辟好空间的指针。
- 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
- 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
- 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
free函数:
我们有可以开辟空间的函数,肯定也有释放空间的函数啊!

?free函数如何使用:

- free函数的返回值是void 。
- 而free的参数是需要一个动态空间的起始地址,它会将这个起始地址之后的动态空间都会被释放
使用的时候存在的一些问题:?
动态空间释放不完全:

?我们这里发现我们释放的是q而不是p,这其中的原因是什么呢!

我们发现了,p的位置在不断向后走,而我们free函数的参数是一块空间的起始地址,然后我们free(p),我们就只释放后面30空间,而前面10个空间并没有释放,就造成了空间的浪费。
所以我们开辟空间的时候可以将开辟空间的起始地址赋值给一个指针,而再将这个指针赋值给一个二级指针,我们利用这个一级指针进行各种操作,而利用二级指针来free释放空间就好了。这样就不会因为不小心而造成空间的浪费了。
当开辟的空间的起始地址为空指针:?

我们不能对空指针进行解引用操作,所以得先判断是否为空指针,避免错误。?
?calloc函数:
calloc如何定义:

?calloc与malloc都是用来开辟动态空间的?
与malloc函数不同的是:
calloc有两个参数:第一个参数:个数;第二个参数:每个变量的字节大小;
calloc开辟的空间大小为个数x每个变量的字节大小;如:5(个数)*4(int类型的大小)
函数,;
calloc如何使用:?

realloc函数:?
realloc与malloc和calloc函数不太一样,realloc函数是用来更改空间大小的函数。
relloc如何定义:.

?realloc函数既然是更改动态空间的大小,它肯定要先找到这一块动态空间!
所以realloc函数有两个参数,
第一个参数:起始地址(已经开辟过的动态空间);
第二个参数:调整过后想要的字节大小。
例如:16变为40,第二个参数就为40.
当我们更改后的动态空间的大小不够时:
- 有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小的调整.
- 如果我们调整大小时,需要扩大了,但后面的空间不够,会重新开辟一个足够大的空间,然后将之前是数据拷贝一份放在前面,然后继续开辟后面的空间,然后返回新的地址.

静态和动态内存的存储地方:
- 我们平常定义的变量存放在栈区,我们程序开始他们就开辟空间,我们程序结束他们释放了
- 动态内存分配开辟在堆区如果我们动态内存开辟后,如果我们程序没有结束它就一种占用着,除非你自己将他释放了(利用free函数)。但程序结束后他们也释放了
?
动态内存分配常见错误:
当malloc开辟的空间的起始地址为空指针:

?我们不能对空指针进行解引用操作,所以得先判断是否为空指针,避免错误。
动态内存越界访问的问题:

这里我们最后会指向p+20,这里会有产生越界访问?

?对非动态开辟内存使用free释放:
?会报错的,别犯贱这么写

?使用free释放一块动态开辟内存的一部分:
这个就是为什么我们要提前用一个指针将起始地址保存好的原因

对同一块动态内存多次释放?:
我们不能对同一块动态内存进行多次释放,会报错的
我们释放完后将这个地址设为空指针,可以避免错误
?动态开辟内存忘记释放(内存泄漏):
忘记释放不再使用的动态开辟的空间会造成内存泄漏。
切记:
动态开辟的空间一定要释放,并且正确释放 。
?
关于动态内存相关的笔试题:

这里str指向了p,而在函数中开设 了空间,并将起始地址给了p,p的地址以及变成了0x11223344,而p为形参不能对实参进行影响,所以str还是空指针,我们不能对空指针进行解引用。?

C和C++的内存开辟:
C/C++程序内存分配的几个区域:
1. 栈区(stack):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。
2. 堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配方式类似于链表。
3. 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
4. 代码段:存放函数体(类成员函数和全局函数)的二进制代码。

?柔性数组:
在C99标准下才可以使用
柔性数组使用的注意事项?:
- 结构中的柔性数组成员前面必须至少一个其他成员。
- sizeof 返回的这种结构大小不包括柔性数组的内存。
- 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。
?
柔性数组的使用?:

柔性数组相对于自定义结构的好处:
第一个好处是:方便内存释放
第二个好处是:这样有利于访问速度.
?
|