使用malloc和free有哪些注意事项?
2个回答
推荐于2016-06-21 · 知道合伙人宠物行家
关注
展开全部
free 不管指针指向多大的空间,均可以正确地进行释放,这一点释放比 delete/delete [] 要方便。不过,必须注意,如果在分配指针时,用的是new或new[]。当在释放内存时,并不能图方便而使用free来释放。反过来,用malloc 分配的内存,也不能用delete/delete[] 来释放。一句话,new/delete、new[]/delete[]、malloc/free 三对均需配套使用,不可混用.
1 分配内存 malloc 函数
需要包含头文件:
and
函数声明(函数原型)
void *malloc(int size);
说明:malloc 向系统申请分配指定size个字节的内存空间。返回类型是 void* 类型。void* 表示未确定类型的指针。C,C++规定,void* 类型可以强制转换为任何其它类型的指针。
从函数声明上可以看出。malloc 和 new 至少有两个不同: new 返回指定类型的指针,并且可以自动计算所需要大小。比如:
int *p;
p = new int; //返回类型为int* 类型(整数型指针),分配大小为 sizeof(int);
或:
int* parr;
parr = new int [100]; //返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int) * 100;
而 malloc 则必须由我们计算要字节数,并且在返回后强行转换为实际类型的指针。
int* p;
p = (int *) malloc (sizeof(int));
第一、malloc 函数返回的是 void * 类型,如果你写成:p = malloc (sizeof(int)); 则程序无法通过编译,报错:“不能将 void* 赋值给 int * 类型变量”。所以必须通过 (int *) 来将强制转换。
第二、函数的实参为 sizeof(int) ,用于指明一个整型数据需要的大小。如果你写成:
int* p = (int *) malloc (1);
代码也能通过编译,但事实上只分配了1个字节大小的内存空间,当你往里头存入一个整数,就会有3个字节无家可归,而直接“住进邻居家”!造成的结果是后面的内存中原有数据内容全部被清空。
malloc 也可以达到 new [] 的效果,申请出一段连续的内存,方法无非是指定你所需要内存大小。
比如想分配100个int类型的空间:
int* p = (int *) malloc ( sizeof(int) * 100 ); //分配可以放得下100个整数的内存空间。
另外有一点不能直接看出的区别是,malloc 只管分配内存,并不能对所得的内存进行初始化,所以得到的一片新内存中,其值将是随机的。
除了分配及最后释放的方法不一样以外,通过malloc或new得到指针,在其它操作上保持一致。
2 释放内存 free 函数
需要包含头文件(和 malloc 一样):
函数声明:
void free(void *block);
即: void free(指针变量);
之所以把形参中的指针声明为 void* ,是因为free必须可以释放任意类型的指针,而任意类型的指针都可以转换为void *。
举例:
int* p = (int *) malloc(4);
*p = 100;
free(p); //释放 p 所指的内存空间
或者:
int* p = (int *) malloc ( sizeof(int) * 100 ); //分配可以放得下100个整数的内存空间。
……
free(p);
1 分配内存 malloc 函数
需要包含头文件:
and
函数声明(函数原型)
void *malloc(int size);
说明:malloc 向系统申请分配指定size个字节的内存空间。返回类型是 void* 类型。void* 表示未确定类型的指针。C,C++规定,void* 类型可以强制转换为任何其它类型的指针。
从函数声明上可以看出。malloc 和 new 至少有两个不同: new 返回指定类型的指针,并且可以自动计算所需要大小。比如:
int *p;
p = new int; //返回类型为int* 类型(整数型指针),分配大小为 sizeof(int);
或:
int* parr;
parr = new int [100]; //返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int) * 100;
而 malloc 则必须由我们计算要字节数,并且在返回后强行转换为实际类型的指针。
int* p;
p = (int *) malloc (sizeof(int));
第一、malloc 函数返回的是 void * 类型,如果你写成:p = malloc (sizeof(int)); 则程序无法通过编译,报错:“不能将 void* 赋值给 int * 类型变量”。所以必须通过 (int *) 来将强制转换。
第二、函数的实参为 sizeof(int) ,用于指明一个整型数据需要的大小。如果你写成:
int* p = (int *) malloc (1);
代码也能通过编译,但事实上只分配了1个字节大小的内存空间,当你往里头存入一个整数,就会有3个字节无家可归,而直接“住进邻居家”!造成的结果是后面的内存中原有数据内容全部被清空。
malloc 也可以达到 new [] 的效果,申请出一段连续的内存,方法无非是指定你所需要内存大小。
比如想分配100个int类型的空间:
int* p = (int *) malloc ( sizeof(int) * 100 ); //分配可以放得下100个整数的内存空间。
另外有一点不能直接看出的区别是,malloc 只管分配内存,并不能对所得的内存进行初始化,所以得到的一片新内存中,其值将是随机的。
除了分配及最后释放的方法不一样以外,通过malloc或new得到指针,在其它操作上保持一致。
2 释放内存 free 函数
需要包含头文件(和 malloc 一样):
函数声明:
void free(void *block);
即: void free(指针变量);
之所以把形参中的指针声明为 void* ,是因为free必须可以释放任意类型的指针,而任意类型的指针都可以转换为void *。
举例:
int* p = (int *) malloc(4);
*p = 100;
free(p); //释放 p 所指的内存空间
或者:
int* p = (int *) malloc ( sizeof(int) * 100 ); //分配可以放得下100个整数的内存空间。
……
free(p);
AiPPT
2024-12-03 广告
2024-12-03 广告
作为北京饼干科技有限公司的一员,对于市场上各类工具都有所了解。就AiPPT而言,它确实为用户提供了便捷高效的PPT制作体验。通过智能化的辅助功能,用户能够快速生成专业且富有创意的演示文稿,极大地节省了时间和精力。无论是对于个人用户还是企业团...
点击进入详情页
本回答由AiPPT提供
2013-07-08
展开全部
深入理解malloc 和free (by slowaction@venus )
一直对linux 上面内存的获取和释放感兴趣,偷空草草看了看glibc/ptmalloc2 的代码, 有了点粗浅的认识,写出来博大家一笑,不当之处,请大家批评。本文不打算从代码细节上着手,只是分析一下ptmalloc2 算法的基本流程和使用的注意事项。( 这个是提纲,逐步丰富内容吧)
[separator]
用户态这个算作上篇,日后写一点对内核相关系统调用的理解,算下篇吧。
首先明确malloc 获取内存和free 释放内存都是glibc 函数。我们说的获取和释放都是和Glibc 交互,至于是否直接导致glibc 和操作系统交互,由glibc 自己决定.Glibc 有自己的内存申请和释放的算法,就是ptmalloc2 的算法。
malloc(size_t bytes)
1, 取得malloc 区域的锁
Malloc 是个线程安全的函数,避免多线程同时malloc 造成混乱,先要取得区域的锁。 具体的处理机制是,首先trylock main heap ,失败的话,就会尝试建立一个新的heap( 使用mmap), 这样做可以保证多线程的malloc 有很好的响应速度,相应的问题就是,会建立很多的heap, 对内核来说过多的线性区会寻址复杂。 所以最好不要多线程同时频繁malloc.
2 ,对bytes 做一下预处理
加上控制头的大小并且向8 字节对齐。就是说你malloc(1 )和malloc(3) 是一样的。 对字节数不用太计较,glibc 会做的
3 ,如果bytes 小于max_fast, 直接从fastbins (单链表)中间分配,默认是64 字节 . 分配64 以下是不会造成内存碎片的,不用担心
4 ,64-512 的大小,这个区间是按照8 字节的步长排好的,直接从链表(双链表)摘就可以了。
5 ,如果从快速的链表分配失败了。尝试从大块数据的链表上切一小块下来给我们,剩下的存起来下次用。(这个和内核的伙伴算法很像) 分配过多的小空间还是有点麻烦,呵呵
6 ,如果是>512 的要求从largebin 分配了,这个步长要大很多了,是最佳适应的算法,剩下的空间要挂到上面的双向链表里面。
7 .128k -1M, 对于glibc 来说这个实在是很大的要求了,直接从未分配的空间取下一块了
8 ,以上条件都没满足,或者说没完成的话就要动用系统调用了. 如果>1M 的话(当然还有其他的限制条件)直接mmap 开一块heap 区域。
9, 小于1M 的要求,就要sbrk 扩张heap 区域,扩张失败的话,在开辟一块heap 区域,当然了我们会尝试和并这两个heap 区域做集中管理。对富裕的有可能还要free 掉。
看完了malloc,free 就好理解多了
1 ,<64, 直接挂到fastbins
2, 其他的都挂到保留空间
3 ,主heap 的>128K ,释放掉,非主heap 的>64k 就释放,因为主heap 是肯定存在的,其他heap 可能在缩小的过程中就整体释放了
4 ,mmap 的空间,没什么商量的,释放掉。
总结一下:
<64字节的从cache分配,极快。64-512的从cache分配,很快。512-128K,FIFO的算法,稍慢一点。
128K-1M brk系统调用,慢
>1M mmap系统调用,很慢,并且以后寻址有负担。
所以,512以下没有必要用内存池(这里还有个64K的释放基线暂且不提)
512-128K 建议用
128K以上的内存分配强烈推荐使用内存池自己管理。
glibc是没有办法是放中间的“空洞“的,要注意处理内存的顺序。
一直对linux 上面内存的获取和释放感兴趣,偷空草草看了看glibc/ptmalloc2 的代码, 有了点粗浅的认识,写出来博大家一笑,不当之处,请大家批评。本文不打算从代码细节上着手,只是分析一下ptmalloc2 算法的基本流程和使用的注意事项。( 这个是提纲,逐步丰富内容吧)
[separator]
用户态这个算作上篇,日后写一点对内核相关系统调用的理解,算下篇吧。
首先明确malloc 获取内存和free 释放内存都是glibc 函数。我们说的获取和释放都是和Glibc 交互,至于是否直接导致glibc 和操作系统交互,由glibc 自己决定.Glibc 有自己的内存申请和释放的算法,就是ptmalloc2 的算法。
malloc(size_t bytes)
1, 取得malloc 区域的锁
Malloc 是个线程安全的函数,避免多线程同时malloc 造成混乱,先要取得区域的锁。 具体的处理机制是,首先trylock main heap ,失败的话,就会尝试建立一个新的heap( 使用mmap), 这样做可以保证多线程的malloc 有很好的响应速度,相应的问题就是,会建立很多的heap, 对内核来说过多的线性区会寻址复杂。 所以最好不要多线程同时频繁malloc.
2 ,对bytes 做一下预处理
加上控制头的大小并且向8 字节对齐。就是说你malloc(1 )和malloc(3) 是一样的。 对字节数不用太计较,glibc 会做的
3 ,如果bytes 小于max_fast, 直接从fastbins (单链表)中间分配,默认是64 字节 . 分配64 以下是不会造成内存碎片的,不用担心
4 ,64-512 的大小,这个区间是按照8 字节的步长排好的,直接从链表(双链表)摘就可以了。
5 ,如果从快速的链表分配失败了。尝试从大块数据的链表上切一小块下来给我们,剩下的存起来下次用。(这个和内核的伙伴算法很像) 分配过多的小空间还是有点麻烦,呵呵
6 ,如果是>512 的要求从largebin 分配了,这个步长要大很多了,是最佳适应的算法,剩下的空间要挂到上面的双向链表里面。
7 .128k -1M, 对于glibc 来说这个实在是很大的要求了,直接从未分配的空间取下一块了
8 ,以上条件都没满足,或者说没完成的话就要动用系统调用了. 如果>1M 的话(当然还有其他的限制条件)直接mmap 开一块heap 区域。
9, 小于1M 的要求,就要sbrk 扩张heap 区域,扩张失败的话,在开辟一块heap 区域,当然了我们会尝试和并这两个heap 区域做集中管理。对富裕的有可能还要free 掉。
看完了malloc,free 就好理解多了
1 ,<64, 直接挂到fastbins
2, 其他的都挂到保留空间
3 ,主heap 的>128K ,释放掉,非主heap 的>64k 就释放,因为主heap 是肯定存在的,其他heap 可能在缩小的过程中就整体释放了
4 ,mmap 的空间,没什么商量的,释放掉。
总结一下:
<64字节的从cache分配,极快。64-512的从cache分配,很快。512-128K,FIFO的算法,稍慢一点。
128K-1M brk系统调用,慢
>1M mmap系统调用,很慢,并且以后寻址有负担。
所以,512以下没有必要用内存池(这里还有个64K的释放基线暂且不提)
512-128K 建议用
128K以上的内存分配强烈推荐使用内存池自己管理。
glibc是没有办法是放中间的“空洞“的,要注意处理内存的顺序。
本回答被网友采纳
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询