glibc 内存申请和释放及堆连续检查
C语言有两种内存申请方式:
1、静态申请:当你声明全局或静态变量的时候,会用到静态申请内存。静态申请的内存有固定的空间大小。空间只在程序开始的时候申请一次,并且不再释放(除非程序结束)。
2、自动申请:当你声明自动变量的时候会使用自动申请。函数参数、局部变量都属于自动变量。这些变量空间在程序执行致相关语句块申请,离开语句块时释放。
还有一种内存申请方式:动态内存申请。C语言变量并不支持动态内存申请,这一功能由库函数实现。C里面没有动态这个存储类型!!
当你需要存储一些数据,但又不知道这些数据有多大或者这些数据会不断的增加。这是后使用动态申请内存技术会非常有用。动态申请是一种显示申请行为,你必须调用相关函数或者宏,并指名申请的大小。申请到的内存空间只能使用指针访问,动态内存还会消耗额外的计算时间,使用起来颇为不便。所以除非必须自动申请和静态申请都不管用,才会考虑动态申请。当然啦,动态申请是一项很重要的技术,没了他不行。
malloc函数原型:
void *malloc(size_t size)
说明: malloc函数申请的内存空间并不会初始化。我们需要手动使用memset函数将内存快置零。
内存对齐:malloc申请到的内存在32位系统中以8的倍数对齐,64位系统中以16的倍数对齐。如果想以一整页对齐,则需要aligned_alloc或者posix_memalign函数。
返回值:无法申请内存时,返回空指针。
memset函数原型:
void *memset(void *block, int c, size_t size)
如果你想为字符串申请内存,记得大小为长度+1——字符串的最后的空字符并没有被计入长度。
char *ptr;
...
ptr = (char *) malloc (length+); //强制转换
内存释放函数(用来释放malloc申请的内存):
#include<stdlib.h>
void free(void *ptr); //释放ptr指向的内存块
void cfree(void *ptr); //和free一样,但提供与SunOS的兼容性。推荐使用free
如果你想调整动态申请的内存的大小,使用realloc函数,原型如下:
#include<stdlib.h>
void *realloc(void *ptr, size_t newsize)
/*如果旧的内存块后面没有足够的空间申请额外的内存,realloc会重新找一块
*地方申请newsize大小的内存,并将旧内存复制到新内存中。
*返回值:无法申请空间时返回空指针,原内存空间无变化。
*/
申请一块干净的内存空间,我们之前使用malloc,在使用memset清空内存空间。现在我们使用calloc一步到位。
#include <stdlib.h>
void *calloc(size_t count, size_t eltsize)
/*等同于
void * calloc(size_t count, size_t eltsize)
{
size_t size = count * eltsize;
void *value = malloc(size);
if (value != 0) memset(value, 0, size);
return value;
}
*/
malloc的效率问题:
GNU C库中,malloc实际申请的内存大小不会以2的n次方呈现,他会和周围的内存碎片合并,从而避免内存的浪费。
如果需要申请大的内存块(大于一个页),可以使用mmap。mmap函数申请的内存实际上是内存IO映射。我们知道程序面对的是虚拟内存,虚拟内存的背后要有实际物理内存。实际物理内存可以借用辅存。内存是按页划分的,申请大于一个页的内存会有什么事情发生呢?这一页的实际物理内存会被系统锁定。那么很大的物理内存无法被其他程序利用。而我们使用mmap函数,是将虚拟内存和辅存上的文件对应,最重要的是:只有程序对对这块虚拟内存操作的时候,才会申请相应大小实际物理内存。还有个问题是碎片化,如果你申请的内存恰好在两个被使用的内存之间,而你申请的内存有很大,那剩下的空间就很小(可能谁也用不了)(个人理解)
关于内存对齐
malloc 和realloc返回的块地址永远是8的倍数或者64位系统下16的倍数(这和之前所说的大小并不是一回事),如果你想返回更高的倍数可以使用aligned_alloc或者posix_memalign(不过多大的倍数,都必须是2的n次方)
#include<stdlib.h>
void *aligned_alloc(size_t alignment, size_t size)
/*说明*/
//申请size大小的内存块,地址为alignment的倍数(size必须为alignment的倍数)
//如果出错会设置errno,详情请man aligned_alloc
//ISO C11之后的标准支持该函数,在非POSIX系统中兼容性比posix_memalign好
#include <stdlib.h>
void *memalign(size_t boundary, size_t size)
/*说明*/
//memalign同样申请size大小的内存,返回的地址为boundary的倍数。不同的是memalign实际申请的内存大小会稍微大一些(避免内存碎片化),返回的地址实际为内存块里的一个特定边界
#include<stdlib.h>
int posix_memalign(void **memptr, size_t alignment, size_t size)
/*说明*/
//posix_memalign通过memptr返回地址,所以memptr使用了指针的指针。(应该不难理解)
//与memalign不同的地方:alignment的值必须是sizeof(void *)的正数倍,且为2的n次方。(当然所有的内存对齐申请函数的alignment都必须是2的n次方)
#include<stdlib.h>
void *valloc(size_t size) /*说明*/
//等同于如下的函数
void * valloc (size_t size)
{
return memalign(getpagesize(), size);
} //可见,valloc申请的内存边界是以页为大小的。
想要设置内存申请的参数,可以使用mallopt函数
#include <malloc.h>
int mallopt(int param, int value);
传进的参数是一对的,param=value。详细介绍以后再写,或者查看man手册。
动态申请的内存块是堆内存,堆内存不一定是连续的。如果我们想检查其连续性可以使用mcheck函数
#include<mcheck.h>
int mcheck(void (*abortfn) (enum mcheck_status status))
/*简要说明*/
//参数:函数指针,该函数指针指向的函数需要接收mcheck_status的枚举类型。
//mcheck 在检查堆连续性时,如果发现不连续,会调用abortfn指向的函数。如果你提供的时空指针,则会使用默认函数(打印相关信息后停止程序)
//为了能让mcheck函数起作用,你需要在编译选项中添加-lmcheck选项。
//另外你也可以在gdb调试时调用mcheck函数。
//状态值:描述了不连续的类型(具体的查看手册)
glibc 内存申请和释放及堆连续检查的更多相关文章
- Unity内存申请和释放
转自:http://www.jianshu.com/p/b37ee8cea04c 1.资源类型 GameObject, Transform, Mesh, Texture, Material, Shad ...
- C语言学习之我见-malloc和free内存申请及释放函数
malloc函数负责向计算机申请确定大小的内存空间. free函数负责释放malloc的申请空间. (1)函数原型 void free(void *_Memory); void * malloc(si ...
- C语言动态内存的申请和释放
什么是动态内存的申请和释放? 当程序运行到需要一个动态分配的变量时,必须向系统申请取得堆中的一块所需大小的存储空间,用于存储该变量.当不再使用该变量时,也就是它的生命结束时,要显式释放它所占用的存储空 ...
- [CareerCup] 13.9 Aligned Malloc and Free Function 写一对申请和释放内存函数
13.9 Write an aligned malloc and free function that supports allocating memory such that the memory ...
- Win3内存管理之私有内存跟共享内存的申请与释放
Win3内存管理之私有内存跟共享内存的申请与释放 一丶内存简介私有内存申请 通过上一篇文章.我们理解了虚拟内存与物理内存的区别. 那么我们有API事专门申请虚拟内存与物理内存的. 有私有内存跟共享内存 ...
- C++函数中,两个自动释放内存的动态内存申请类
最近做一个事情,实现一个流程交互,其中主交互流程函数中,涉及较多的内存申请, 而健康的函数,都是在函数退出前将手动申请不再需要的内存释放掉, 使用很多方法,都避免不了较多的出错分支时,一堆的if fr ...
- 内存管理概述、内存分配与释放、地址映射机制(mm_struct, vm_area_struct)、malloc/free 的实现
http://blog.csdn.net/pi9nc/article/details/23334659 注:本分类下文章大多整理自<深入分析linux内核源代码>一书,另有参考其他一些资料 ...
- glibc内存管理那些事儿
本文转载自glibc内存管理那些事儿 Linux内存空间简介 32位Linux平台下进程虚拟地址空间分布如下图: 进程虚拟地址空间分布 图中,0xC0000000开始的最高1G空间是内核地址空间,剩下 ...
- 2万字|30张图带你领略glibc内存管理精髓(因为OOM导致了上千万损失)
前言 大家好,我是雨乐. 5年前,在上家公司的时候,因为进程OOM造成了上千万的损失,当时用了一个月的时间来分析glibc源码,最终将问题彻底解决. 最近在逛知乎的时候,发现不少人有对malloc/f ...
随机推荐
- jquery插件课程2 放大镜、多文件上传和在线编辑器插件如何使用
jquery插件课程2 放大镜.多文件上传和在线编辑器插件如何使用 一.总结 一句话总结:插件使用真的还是比较简单的,引包,初始化,配置参数(json),配置数据(json),而后两步不是必须的.而且 ...
- js,jquery遍历数组,对象
each的用法 1.数组中的each 复制代码 var arr = [ "one", "two", "three", "four ...
- Android 比较两个时间段是否有交集或重复
先看一个例图: 在金山<电池管家>应用中就有一个类似上图这样的功能—— 开启多个定时任务. 当开启另一个定时任务的时候,如果即将开启的这个定时任务的时间段与已经开启了的定时任务的时间段有交 ...
- 【前端统计图】hcharts实现堆叠柱形图(与后台数据交互)
原型图类似如下: 图片.png <!DOCTYPE > <html> <head> <meta charset="utf-8">&l ...
- Qt5.4.2编译 qtpropertybrowser-2.51(从4修改到5的除错过程)
qtpropertybrowser是一个洛基亚时代的一个产物,其实它就是QtDesigner侧栏那种样子而已.网上大部分的都只适用于Qt4,今天头脑发热想用Qt5编译一下,发觉一大堆错误,因为Qt4有 ...
- 真机测试时出现 could not find developer disk image问题
解决Xcode在ipad/iphone 系统真机测试时出现could not find developer disk image问题 原因:手机系统版本比xcode版本高,sdk不支持 方法:更新Xc ...
- Java设计模式菜鸟系列(二十二)中介者模式建模与实现
转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/40027109 中介者模式(Mediator):主要用来减少类与类之间的耦合的,由于假设类与 ...
- 窗体背景的绘制(Windows窗体每次都会重绘其窗体背景,所以我们可以通过拦截窗体重绘背景的消息(WM_ERASEBKGND),并自定义方法来实现重绘窗体背景)
核心思想:由于Windows窗体每次都会重绘其窗体背景,所以我们可以通过拦截窗体重绘背景的消息(WM_ERASEBKGND),并自定义方法来实现重绘窗体背景.通过TImage组件也可以实现,但是重写W ...
- Android四大组件之Activity--管理方式
1. 概览 Activity的管理有静态和动态两层涵义: 静态是指Activity的代码组织结构,即Application中声明的Activity的集合,这些Activity被组织在一个APK中,有特 ...
- tensorflow 函数接口的理解
1. tf.nn.softmax tf.nn.softmax(logits, dim=-1, name=None) w*x+b ⇒ logits softmax 函数执行的操作:exp(logits) ...