【C语言】内存分配函数malloc/ calloc/ realloc及内存释放free

前言

内存区域划分与分配:

1、栈区(stack)——程序运行时由编译器自动分配,存放函数的参数值,局部变量的值等,程序结束时由编译器自动释放。

2、堆区(heap) —— 在内存开辟另一块存储区域。一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。用malloc, calloc, realloc等分配内存的函数分配得到的就是在堆上。

3、全局区(静态区)(static)——编译器编译时即分配内存。全局变量和静态变量的存储是放在一块的。对于C语言初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。而C++则没有这个区别 - 程序结束后由系统释放

4、文字常量区 ——常量字符串就是放在这里的, 程序结束后由系统释放。

5、程序代码区——存放函数体的二进制代码。


初识内存分配函数malloc/ calloc/ realloc及内存释放free

从前言中我们知道,如果要想申请一段动态内存需要常见的堆上内存管理函数malloc(), calloc(), recalloc(), free(),而堆上的内存空间不会自动释放,直到调用free()函数,才会释放堆上的存储空间。

接下来我们一起来看一下这些函数具体实现及功能区别

1、malloc()

用于申请指定大小的存储空间。

头文件:stdlib.h

声明:void *malloc(size_t size);

含义:在堆上,分配size个字节,并返回void指针类型(使用该函数申请空间时,需要将空间类型强转为目标类型)。

返回值:分配内存成功,返回分配的堆上存储空间的首地址;否则,返回NULL

例如:给一个长度为4的字符数组申请一个动态内存

char *p = (char *)malloc(4*sizeof(char));

2、calloc()

头文件:stdlib.h

声明:void *calloc(size_t nobj, size_t size);

含义:在堆上,分配nobj*size个字节,并初始化为0,返回void* 类型

返回值:同malloc() 函数

例如:给一个长度为4的字符数组申请一个动态内存,并初识化为0.

char *p = (char *)calloc(4,sizeof(char));

malloc的等价使用如下:

char *p = (char *)malloc(4*sizeof(char));

for(int i=0;i<4;i++)

{

p[i]=0;

由上述实例可知:calloc函数更加全面,经过calloc函数申请的空间都已被初始化为0,而malloc函数申请的空间未被初始化,存储单元存储的数据大小未知。而且calloc函数申请空间时更加方便,第一个参数为数值容量,第二个参数为数组元素的空间大小。

3、recalloc()

头文件:stdlib.h

声明:void *realloc(void *p, size_t size);//新的大小一定要大于原来的大小不然的话会导致数据丢失!

含义:使指针p指向的空间大小变为size,如果p原来指向的空间大小小于size,则系统将试图合并p与其后的空间,若能满足需求,则指针指向不变;如果不能满足,则系统重新为p分配一块size大小的空间,同时会复制原有内容到新分配的存储空间。注意,若原来的void指针p在堆上的空间不大于n个字节,则保持不变。

返回值:同malloc() 函数

例如:给一个长度为4的字符数组动态内存扩容为一个长度为8的字符数组

 p=(char*)realloc(p,8*sizeof(char))

 

malloc函数的等价实现:

char *p = (char *)malloc(4*sizeof(char));

for(int i=0;i<4;i++)

{

p[i]=i;

char *q = (char *)malloc(8*sizeof(char));

for(int i=0;i<4;i++)

{

q[i]=p[i];

free(p);

p=q;

q=NULL;

(过程大概如右:重新申请一段内存——>将原来的值移到新空间——>释放原来内存——>更改地址)

使用malloc、calloc、realloc函数申请的空间都为堆空间,程序结束之后,系统不会将其自动释放,需要程序员自主管理。

4、free()

头文件:stdlib.h

声明:void free(void *p);

含义:释放void指针p所指的堆上的空间。

返回值:无

对于free(p)这句语句,如果p 是NULL 指针,那么free 对p无论操作多少次都不会出问题。如果p 不是NULL 指针,那么free 对p连续操作两次就会导致程序运行错误。

free函数如何知道释放的位置及大小?

答案:实际上在free释放的时候会根据传入的地址向前偏移一定字节(头), 从这些字节中获取具体的内存块大小并释放。

free 函数崩溃的原因:

1、越界(把尾信息给破坏了)

2、修改指针的指向(free找不到头信息)

3、重复释放同一段内存

4、释放非动态创建的内存

若用户申请的堆空间没有及时回收,可能会导致内存泄漏。若发生内存泄漏,则某个进程可能会逐渐占用系统可提供进程的存储空间,该进程运行时间越长,占用的存储空间就越多,知道最后耗尽全部存储空间,导致系统崩溃。

内存泄漏是从操作系统的角度考虑的,这里的存储空间并非指物理内存,而是指虚拟内存大小,这个虚拟内存的大小取决于磁盘交换区设定的大小。由程序申请一块内存,如果没有指针指向它,那么就说明这块内存泄漏了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值