大家好!今天小编给大家带来的是C语言动态内存管理的详细介绍,一起来看看吧!
1.为什么要动态内存管理?
1.1前提
int n = 10;
char c = 'w';
以上有以下特点:
- 一旦开辟空间,大小就固定了
- 数组在声明的时候,必须指定数组长度,数组空间一旦确定大小就不能改变了。
- C语⾔引⼊了动态内存开辟,让程序员⾃⼰可以申请和释放空间,就⽐较灵活了。
1.2学习内容
学习重点:4个函数:malloc,free,calloc,realloc(非常重要)
2.malloc和free
前提:malloc和free都声明在stdlib.h头文件中
2.1 malloc
malloc是C语言中一个动态内存开辟的函数
void* malloc (size_t size);//参数是大小,单位:字节
这个函数向内存申请⼀块连续可⽤的空间,并返回指向这块空间的指针。
- 如果开辟成功,返回一个指向开辟好空间的指针。
- 如果开辟失败,返回一个NULL指针,因此malloc的返回值一定要检查。
- 返回值的类型是void*,开辟空间的类型具体由使用者自己而定。
- 如果参数 size 为0,malloc的⾏为是标准是未定义的,取决于编译器。
int* p = (int*)malloc(20);
//向内存申请20个字节-5个整数放在数组中
2.2 free
free是专门用来做动态内存的释放和回收的
void free (void* ptr);
注意:free是用来释放动态开辟的内存的
- 如果参数ptr指向的空间不是动态开辟的内存,则free的行为无效。
- 如果参数 ptr 是NULL指针,则函数什么事都不做。
举例:
#include <stdio.h>
#include <stdlib.h>
int main()
{
//20 个字节 - 存放5个整数
int* p = (int*)malloc(20);
if (p == NULL)//检查是不是空指针
{
perror("malloc");
return 1;
}
//使用空间
int i = 0;
for (i = 0; i < 5; i++)
{
*(p + i) = i + 1;
}
//释放内存
free(p);//传递给free函数的是要释放的内存空间的起始地址
p = NULL;//拴在树上
return 0;
}
3.calloc和realloc
3.1 calloc
C语言中也提供了calloc函数来实现动态内存分配。
void* calloc (size_t num, size_t size);
两个参数:数量和单价,即num 和 size
理解:
1. 函数的功能是为 num 个⼤⼩为 size 的元素开辟⼀块空间,并且把空间的每个字节初始化为0。
2. 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。
示例:
int main()
{
int* p = (int*)calloc(5, sizeof(int));
if (p == NULL)
{
perror("calloc");
return 1;
}
int i = 0;
for (i = 0; i < 5; i++)
{
printf("%d ", *(p + i));//遍历每个元素
}
free(p);
p = NULL;
return 0;
}
所以,当我们需要对申请的内存空间初始化就可以使用calloc函数,很方便。
3.2 realloc
3.2.1背景
- 有时候我们会发现申请的内存空间的大小太小了或太大了,为了合理使用内存,我们需要对内存空间进行合理的调整。
- C语言提供了realloc函数来进行动态内存开辟空间的大小调整。
3.2.2函数原型
void* realloc (void* ptr, size_t size);
理解:
- ptr是要调整内存的地址
- size是调整后新大小
- 返回值为调整后的内存起始地址
- 这个函数调整原内存空间⼤⼩的基础上,还会将原来内存中的数据移动到 新 的空间。
- realloc在调整内存空间的是存在两种情况:
原有空间 有足够大的空间 没有足够大的空间
3.2.2两种情况
- 原内存有足够大的空间:原空间数据不动
- 原内存没有足够大的空间:在堆空间上另找⼀个合适⼤⼩
的连续空间来使⽤。这样函数返回的是⼀个新的内存地址。
int main()
{
int* p = (int*)malloc(20);
//希望将空间调整为40个字节
int *ptr = (int*)realloc(p, 40);
if (ptr == NULL)
{
perror("realloc");
return 1;
}
p = ptr;//不是空指针,则放回到p中维护
//使用
int i = 0;
for (i = 0; i < 10; i++)
{
*(p + i) = i + 1;
}
for (i = 0; i < 10; i++)
{
printf("%d ", *(p + i));
}
free(p);
p = NULL;
return 0;
}
4.常见动态内存的错误
4.1对NULL指针的解引用操作
void test()
{
int *p = (int *)malloc(INT_MAX/4);
*p = 20;//如果p的值是NULL,就会有问题
free(p);
}
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);
}
4.3对⾮动态开辟内存使⽤free释放
void test()
{
int a = 10;
int* p = &a;
free(p);//error
}
4.4使⽤free释放⼀块动态开辟内存的⼀部分
void test()
{
int *p = (int *)malloc(100);
p++;
free(p);//p不再指向动态内存的起始位置
}
4.5对同⼀块动态内存多次释放
void test()
{
int *p = (int *)malloc(100);
free(p);
free(p);//重复释放
}
4.6动态开辟内存忘记释放(内存泄漏)
void test()
{
int *p = (int *)malloc(100);
if(NULL != p)
{
*p = 20;
}
}
int main()
{
test();
while(1);
}
5.总结
C语言动态内存管理的内容对后续学习和操作至关重要,重点把握几个函数的使用方法,以及动态内存的内涵!