目录
1、为什么需要动态的内存分配方式?
当前我们建立变量如
int n = 100;
char arr[100]={0};
都是在栈区开辟相应的内存空间存放数据。此时对变量来说开辟的空间大小是确定的,对数组来说必须要有确定的元素数量,在编译时才能在内存中分配空间。一旦程序开始运行,分配的空间都是固定的。
但是如果我们需要在程序运行的时候改变变量的存储空间,因为有时我们需要的空间大小在程序运行的时候才能知道, 那数组的编译时开辟空间的方式就不能满足了。此时就需要用在堆区开辟动态内存。
2、动态内存函数的介绍
2.1、malloc() and free()
C语言提供了malloc一个动态内存开辟的函数:
#include <stdlib.h> //需要头文件
void* malloc (size_t size);
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
a: 如果开辟成功,则返回一个指向开辟好空间的指针。
b: 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
c: 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己 来决定。
d: 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
C语言提供了另外一个函数free,专门是用来做动态内存的释放和回收的,函数原型如下:
#include <stdlib.h> //需要头文件
void free (void* ptr);
free函数用来释放动态开辟的内存。
a: 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
b: 如果参数 ptr 是NULL指针,则函数什么事都不做。
使用实例:
#include <stdio.h>
int main()
{
//代码1 为在栈区开辟的空间
int num = 0;
scanf("%d", &num);
int arr[num] = {0};
//代码2 为在堆区开辟的空间
int* ptr = NULL;
ptr = (int*)malloc(num*sizeof(int));
if(NULL != ptr)//判断ptr指针是否为空,不为空然后使用其中的空间
{
int i = 0;
for(i=0; i<num; i++)
{
*(ptr+i) = 0;
}
}
free(ptr);//使用结束后,释放ptr所指向的动态内存
ptr = NULL;//必须要将指针置为空,因为ptr此时指向的空间已被free
//没有权限访问,否则有可能发生野指针情况。
return 0;
}
2.2 calloc
C语言还提供了一个函数叫 calloc , calloc 函数也用来动态内存分配。
#include <stdlib.h>
void* calloc (size_t num, size_t size);
a: 函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。
b: 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。
若申请的内存空间的内容要求初始化,那么可以很方便的使用calloc函数来完成任务。
2.3 realloc
为什么会有realloc
a: realloc函数的出现让动态内存管理更加灵活。
b: 我们一定会对内存的大小做灵活的调整,使其可大可小,满足任务要求,那 realloc 函数就可以做到对动态开辟内存大小的调整。