很多情况下,我们开辟一个静态的内存空间时,会导致不够用的情况发生。那么这时我们就会用到动态内存分配来进行动态内存开辟。动态内存分配函数有:malloc,calloc,free,realloc等函数,这些函数都是在堆区使用的,换句话说,我们静态开辟的内存空间是在栈区,动态开辟的内存空间是在堆区。
1、malloc函数
我们在动态内存分配的时候就是用malloc函数在堆区开辟一个动态内存空间。用法如下:
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
(1)如果开辟成功,则返回一个指向开辟好空间的指针。
(2)如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
(3)malloc函数并不知道开辟空间的类型,具体使用是由我们来自行决定。
(4)malloc函数参数表示要开辟空间的大小,我这代码代表要开辟10个整型大小的空间。如果参数为0时,大小取决于编译器。
当我们使用完动态开辟的内存空间后,一定要记得释放和回收空间,不然会造成内存泄漏,导致不必要的麻烦。
2、free函数
free函数是用来释放动态开辟的内存空间,具体用法如下:
3、calloc函数
当我们要对申请的内存空间的内容要求初始化,那么就可以很方面的使用calloc函数来实现。
具体用法和malloc函数差不多,用法如下:
4、realloc函数
当我们想要调整动态空间大小的时候,可以使用realloc函数,让我们动态管理更加的灵活。具体用法如下:
参数里面,p为要调整的内存地址,后面接着调整之后的新大小。
realloc返回值为调整之后的内存起始位置。还会将原来内存中的数据移动到新空间。
5、动态内存空间开辟常见错误
(1)对NULL指针的解引用操作
万一创建失败的话,就意味着对NULL指针的解引用操作,所以当我们要用malloc函数的时候一定得记着对malloc函数值做判空处理。
(2)对动态开辟空间得越界访问
有时候我们开辟40个字节的空间误认为开辟了40个整型空间,这样就会导致对动态开辟空间的越界访问。
(3)对非动态开辟内存使用free释放
free是释放堆区动态开辟的内存空间,而非动态开辟的内存空间在栈区。
(4)使用free释放一块动态开辟内存的一部分
动态开辟内存空间他是一块连续的空间,是不能一点一点释放的。
(5)对同一块动态开辟的空间多次释放
如果在第一次释放完没有及时手动把p设置为NULL,那么第二次释放是不允许的,释放的空间不能再次释放,所以我们要养成良好的代码习惯。
(6)动态开辟的空间内存忘记释放
这种错误,可谓是致命的,应为它会造成内存泄漏。
6、柔性数组
C99中,结构中的最后一个元素允许是未知大小的数组,这就叫做【柔性数组】
例如:
结构中的柔性数组成员前面必须至少一个其他成员。
sizeof返回的这种结构大小不包括柔性数组的内存
例如:
此时返回的值是n的大小,而不包括柔性数组的大小。
包含柔性数组成员的结构用malloc函数进行内存的动态分配,并且分配的内存应该大于结构的大小以适应柔性数组的语气大小。
sizeof(struct S)是结构体成员的大小(不包含柔性数组)。