为什么要有动态内存
例如 int a=20 char arr[10]={10};
这些的内存是固定的
但是有时候我们需要的空间大小是在程序运行的时候才能知道的
malloc和free
void* malloc (size_t size)
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
如果开辟成功,则返回一个指向开辟好空间的指针。 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查
返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己 来决定
void free (void* ptr);
拿来释放动态内存的
举个例子
#include <stdio.h>
int main()
{//代码1int num = 0;scanf("%d", &num);int arr[num] = {0};//代码2int* 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;
}
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){//使用空间 }free(p);p = NULL;return 0;
}
realloc
他可以调整动态内存开辟的大小
void* realloc (void* ptr, size_t size);
ptr 是要调整的内存地址 size 调整之后新大小
几个常见的错误
对NULL指针的解引用
void test()
{int *p = (int *)malloc(INT_MAX/4);*p = 20;//如果p的值是NULL,就会有问题free(p);
}
对动态开辟空间的越界访问
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);
}
对非动态开辟内存的使用free释放
void test()
{int a = 10;int *p = &a;free(p);//ok?
}
用free释放一块动态开辟内存的一部分
void test()
{int *p = (int *)malloc(100);p++;free(p);//p不再指向动态内存的起始位置
}
多次释放
用free(p)
free(p)释放两次
动 态 开 辟 内 存 忘 记 释 放 ( 内 存 泄 漏 )