为什么会有动态内存分配
int a = 10;
int arr[10] ;
都是固定申请一个空间用来存放变量,空间是固定的,后续无法改变
动态内存管理能够让后续根据实际需求,改变空间大小。
动态内存中分配的函数
malloc
calloc
realloc
free
<stdlib.h>
malloc
void* malloc(size_t size)
申请一块连续可用的内存空间,然后返回这个空间的指针
size是需要开辟的内存空间大小(字节)
由于返回类型是void,所以在使用的时候,要强制类型转换为需要的类型,如下,转换成int*
int main()
{
int *p=(int*)malloc(40);//申请40字节的空间,用于存放10个int类型
return 0;
}
实际使用时,可能会出现申请空间失败的情况
1.如果申请空间成功,返回指向 开辟好空间的 指针。
2.如果申请空间失败,就会返回空指针NULL
因此,使用时一定要检查返回的指针,如下
int main()
{
int *p=(int*)malloc(40);//申请40字节的空间,用于存放10个int类型
if(p==NULL)
{
printf("%s\n",strerror(errno));//如果是空指针,打印错误信息,别忘了引头文件
}
return 0;
}
申请到的空间没有自动初始化。
free
void free( void* prt )
作用:释放申请的内存
当申请了动态内存空间,如果不释放,也就是还回去,那么这个空间就会一直被占用。
只要把需要释放的空间的地址传过去就行。!!一定要注意free只能释放动态开辟的空间,也就是这些动态内存管理函数所开辟的空间。
在使用free时,最好加上
int main()
{
//申请内存的操作看上面的图
free(p);
p=NULL;//通过赋空指针来手动清零
return 0;
}
因为内存空间释放就是把申请的空间还回去,但是那个空间里的内容还存在,为了防止会影响到程序,手动把那个空间的内容清零。
calloc
void* calloc(size_t num,size_t size)
num是元素个数,size是每个元素的大小(字节)
作用是为num个大小为size的元素申请空间,并且还会初始化每个元素为0。
用法和malloc一样。
realloc
void* realloc(void*ptr,size_t size)
作用是调整动态内存空间的大小
申请的动态内存空间是可以变化的,变化就靠这个函数
ptr是要调整大小的空间地址,size是要调整 到的 新的 大小
int main()
{
//申请动态内存空间并检查
int* p = (int*)malloc(5 * sizeof(int));
if (NULL == p)
{
perror("malloc");
return 1;
}
//增加5个整型空间
int *ptr = (int*)realloc(p, 10 * sizeof(int));//调整成10个整型的大小
//不能用p接收,要拿新的指针接收。
if (ptr != NULL)
{
p = ptr;
}
//检验下有没有调整成功,调整成功了才拷贝。
return 0;
}
->用新的指针接收,是为了防止调整失败,返回空指针,如果这里用原来的p接收,就会把p里面的内容给覆盖掉(没调整成功,反而损失了数据)。用新的指针接收后,检查后才执行拷贝。
扩容空间
动态内存常见的错误
1.解引用空指针,对空指针进行操作
在开辟空间时,没有检验开辟空间有没有失败。一定要判断
2.对动态开辟内存的越界访问
3.对非动态开辟内存使用free释放
4.使用free释放动态开辟空间的一部分
比如在使用开辟空间时,代码中有p++,改变了p的地址,最后用free时,程序会挂掉
5.把动态开辟空间释放了两次
如何避免:每次在用free(p)后,都要接上p=NULL,这样再来一次free(p)就不会挂
6.没有回收空间
申请的地址如果没有回收,那就永远不会消失,除了重启才能释放那一部分被占用的内存,所以只要申请了,就一定要记得释放,无法马上释放也要打上注释,要让别人释放这个内存
像这样空间一直被占用,导致可用内存空间减少,就是内存泄漏。