一、 为什么要有动态内存分配
我们已经掌握的内存开辟方式有:
int val = 20;// 在栈空间上开辟四个字节
char arr[10] = {0};// 在栈空间上开辟 10 个字节的连续空间
但是上述的开辟空间的方式有两个特点:
1.空间开辟大小是固定的。
2.数组在申明的时候,必须指定数组的长度,数组空间一旦确定了大小不能调整
但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。 C语言引入了动态内存开辟,让程序员自己可以申请和释放空间,就比较灵活了。
二、 malloc和free
2.1 malloc
C语言提供了一个动态内存开辟的函数:
void* malloc (size_t size);
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。如果开辟成功,则返回一个指向开辟好空间的指针。如果开辟失败,则返回一个 NULL 指针,因此malloc的返回值一定要做检查。 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
2.2 free
C语言提供了另外一个函数free,专门是用来做动态内存的释放和回收的,函数原型如下:
void free (void* ptr);
free函数用来释放动态开辟的内存。如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。如果参数 ptr 是NULL指针,则函数什么事都不做。 malloc和free都声明在 stdlib.h 头文件中。 举个例子:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int num = 0;
scanf("%d", &num);
int arr[num] = { 0 };
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;//是否有必要?
return 0;
}
最后一句 ptr = NULL; 从程序运行角度来说并不是必须的,但在实际开发中推荐保留。因为 free(ptr); 只是释放了指针所指向的动态内存,并不会自动把指针变量本身清空,释放后 ptr 仍然保存着原来的地址,此时它就变成了悬空指针(野指针)。如果后续代码中不小心再次使用这个指针,比如解引用或再次调用 free,就可能导致程序崩溃或出现未定义行为。将指针置为 NULL 可以有效避免误用,因为对 NULL 指针的再次 free 是安全的。
三、calloc和realloc
3.1 calloc
C语言还提供了一个函数叫calloc , calloc 函数也用来动态内存分配。原型如下:
void* calloc (size_t num, size_t size);
函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全 0。 举个例子:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int* p = (int*)calloc(10, sizeof(int));
if (NULL != p)
{
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ", *(p + i));
}
}
free(p);
p = NULL;
return 0;
}
代码中通过 calloc(10, sizeof(int)) 申请空间,与 malloc 不同的是,calloc 在分配内存的同时会把这块内存全部初始化为 0,因此后面的循环打印时,每个元素默认都是 0。程序先判断指针 p 是否为空,避免申请失败时发生非法访问;如果申请成功,就通过指针遍历的方式依次输出数组中的值。最后调用 free(p) 释放动态内存,防止内存泄漏,并把指针置为 NULL,以避免形成悬空指针。
输出结果:
0 0 0 0 0 0 0 0 0 0
所以如果我们对申请的内存空间的内容要求初始化,那么可以很方便的使用calloc函数来完成任务。
3.2 realloc
realloc函数的出现让动态内存管理更加灵活。 有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的使用内存,我们一定会对内存的大小做灵活的调整。那realloc函数就可以做到对动态开辟内存大小的调整。函数原型如下:
void* realloc (void* ptr, size_t size);
ptr 是要调整的内存地址,size 调整之后新大小,返回值为调整之后的内存起始位置。 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间。
realloc在调整内存空间的是存在两种情况:
1.原有空间之后有足够大的空间
2.原有空间之后没有足够大的空间
编辑
1:当是1的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化。
2:当是情况2的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适da'x的连续空间来使⽤。这样函数返回的是一个新的内存地址。
由于上述的两种情况,realloc函数的使用就要注意一些。
#include <stdio.h>
#include <stdlib.h>
int main()
{
int* ptr = (int*)malloc(100);
if (ptr != NULL)
{
//业务处理
}
else
{
return -1;
}
//扩展容量
//代码1 -直接将realloc的返回值放到ptr中
ptr = (int*)realloc(ptr, 1000);//这样可以吗?(如果申请失败会如何?)
//代码2 -先将realloc函数的返回值放在p中,不为NULL,在放ptr中
int* p = NULL;
p = realloc(ptr, 1000);
if (p != NULL)
{
ptr = p;
}
//业务处理
free(ptr);
return 0;
}
这段代码的关键问题在于对 realloc 返回值的处理。代码1中直接写成
ptr = (int*)realloc(ptr, 1000); 在语法上是可以的,但存在风险:如果扩容失败,realloc 会返回 NULL,而原来的内存块并不会被释放。此时已经把原来的地址覆盖掉了,既拿不到新空间,又丢失了旧空间的指针,最终会造成内存泄漏,而且后续再使用 ptr 还可能发生空指针错误。
相比之下,代码2是更安全、更规范的写法。先用一个临时指针 p 接收 realloc 的返回值,如果 p 不为 NULL,说明扩容成功,再把它赋给 ptr;如果为 NULL,原来的 ptr 仍然有效,可以决定是否继续使用或释放。
四、 常见的动态内存的错误
4.1 对NULL指针的解引用操作
void test()
{
int* p = (int*)malloc(INT_MAX / 4);
*p = 20;//如果p的值是NULL,就会有问题free(p);
}
正确做法是在使用指针之前先判断是否申请成功,例如:
int* p = (int*)malloc(INT_MAX / 4);
if (p == NULL) {
// 处理申请失败,返回或报错
return;
}
*p = 20;
free(p);
p = NULL;
4.2 对动态开辟空间的越界访问
void test()
{
int i = 0;
int* p = (int*)malloc(10 * sizeof(int));
if (NULL == p)
{
exit(EXIT_FAILURE);
}
for (i = 0; i <= 10; i++)
{
*(p + i) = i;//当i是10的时候越界访问
}
free(p);
}
for 循环条件写成了 i <= 10,当 i == 10 时,表达式 *(p + i) 会访问到第 11 个元素的位置,已经超出了申请的内存范围,属于未定义行为,可能导致数据破坏或程序崩溃。
4.3 对非动态开辟内存使用free释放
void test()
{
int a = 10;
int* p = &a;
free(p);//ok?
}
free 只能释放由动态内存分配函数(如 malloc、calloc、realloc)申请的内存,而代码中的 p 指向的是局部变量 a 的地址,这块内存位于栈区,不是动态分配的。如果对它调用 free(p),就会产生未定义行为,常见结果是程序崩溃或运行异常。
4.4 使用free释放一块动态开辟内存的一部分
void test()
{
int* p = (int*)malloc(100);
p++;
free(p);//p不再指向动态内存的起始位置
}
free 必须接收最初由 malloc(或 calloc、realloc)返回的那个原始指针值,而代码中在 p++ 之后,p 已经不再指向动态内存块的起始地址,而是指向中间位置。此时调用 free(p) 属于未定义行为,常见结果是程序崩溃或内存管理错误。
4.5 对同一块动态内存多次释放
void test()
{
int* p = (int*)malloc(100);
free(p);
free(p);//重复释放}
free(p); 第一次调用后,这块动态内存已经被释放,指针 p 变成了悬空指针;如果再次对同一个指针调用 free,就会产生未定义行为,常见结果包括程序崩溃、堆结构损坏或安全漏洞。
4.6 动态开辟内存忘记释放(内存泄漏)
void test()
{
int* p = (int*)malloc(100);
if (NULL != p)
{
*p = 20;
}
}
int main()
{
test();
while (1);
}
函数 test 中通过 malloc(100) 动态申请了一块堆内存,但在函数结束前并没有调用 free 进行释放,因此这块内存会一直占用着,无法被回收。尤其是在 main 中又进入了死循环 while (1);,程序不会结束,操作系统也不会帮你回收这块内存,泄漏会一直存在。