《深入理解计算机系统》对动态内存分配器的定义是:维护着一个进程的虚拟存储器区域,成为”堆“,对于每一个进程,操作系统内核维护着一个变量brk(break)指向堆的顶部。
下面给出我关于动态内存分配方面的理解:
动态内存分配器,我的理解就是在程序执行的过程当中,动态的分配或释放虚拟内存。
有了动态内存分配的概念,自然就会有动态存储器的概念。所谓动态存储器,简而言之就是我们进程虚拟内存中的堆。堆在动态内存分配的过程中,会动态的增长或减少。堆在虚拟进程的结构当中是向上增长的,当堆内存不够时,还可以向操作系统请求额外的内存。
动态存储器分配器有两种基本分格:
概念:自动的内存管理:显示地分配堆内存,隐式地释放堆内存。
功能:
垃圾收集器避免了悬挂指针(dangling reference),原因是一个仍然被引用的对象永远不会内存回收并且也不会被认为已经被释放掉了。
垃圾收集器解决了空间泄漏(space leaks)问题,原因是它可以自动释放不再被引用的空间。
寻找并释放这些对象的空间的过程就做垃圾收集(garbage collection)。那么在什么时候会出发垃圾收集动作呢?一般来说整个堆或一部分被填满或者达到某一百分比数值时将被收集。
串行(Serial):也被称为完全停顿(Stop-the-world)。当执行GC时,程序的线程都会被挂起。
并行(Parallel):圾收集工作被分成几部分,这些子部将会在不同的CPU上被同时执行。同时执行会使垃圾收集得更快,但是代价是会增加复杂性和潜在碎片。
并发(Concurrent):一个或者多个垃圾收集任务也可以并发的与应用程序同时执行。通常,一个并发收集器可以并发的执行垃圾收集的大部分工作,但是也会不可避免的引发一个小的停顿。
Android 1.0~2.2,Dalvik虚拟使用的垃圾收集机制有以下特点:
Android 2.3~至今,Dalvik虚拟使用的垃圾收集机制得到了改进
Dalvik虚拟机执行完成一次垃圾收集之后,我们通常可以看到类似以下的日志输出:
D/dalvikvm(9050): GC_CONCURRENT freed 2049K, 65% free 3571K/9991K, external 4703K/5261K, paused 2ms+2ms
在这一行日志中,GC_CONCURRENT表示GC原因,2049K表示总共回收的内存,3571K/9991K表示Java Object Heap统计,即在9991K的Java Object Heap中,有3571K是正在使用的,4703K/5261K表示External Memory统计,即在5261K的External Memory中,有4703K是正在使用的,2ms+2ms表示垃圾收集造成的程序中止时间。
Dalvik虚拟机的内存大体上可以分为Java Object Heap、Bitmap Memory和Native Heap三种。
Java Object Heap是用来分配Java对象的,也就是我们在代码new出来的对象都是位于Java Object Heap上的。
Dalvik虚拟机在启动的时候,可以通过-Xms和-Xmx选项来指定Java Object Heap的最小值和最大值。为了避免Dalvik虚拟机在运行的过程中对Java Object Heap的大小进行调整而影响性能,我们可以通过-Xms和-Xmx选项来将它的最小值和最大值设置为相等。
Java Object Heap的最小和最大默认值为2M和16M,但是手机在出厂时,厂商会根据手机的配置情况来对其进行调整,例如,G1、Droid、Nexus One和Xoom的Java Object Heap的最大值分别为16M、24M、32M 和48M。我们可以通过ActivityManager类的成员函数getMemoryClass来获得Dalvik虚拟机的Java Object Heap的最大值。
ActivityManager类的成员函数getMemoryClass的实现如下所示:
public class ActivityManager { ...... /** * Return the approximate per-application memory class of the current * device. This gives you an idea of how hard a memory limit you should * impose on your application to let the overall system work best. The * returned value is in megabytes; the baseline Android memory class is * 16 (which happens to be the Java heap limit of those devices); some * device with more memory may return 24 or even higher numbers. */ public int getMemoryClass() { return staticGetMemoryClass(); } /** @hide */ static public int staticGetMemoryClass() { // Really brain dead right now -- just take this from the configured // vm heap size, and assume it is in megabytes and thus ends with "m". String vmHeapSize = SystemProperties.get("dalvik.vm.heapsize", "16m"); return Integer.parseInt(vmHeapSize.substring(0, vmHeapSize.length()-1)); } ...... }
Dalvik虚拟机在启动的时候,就是通过读取系统属性dalvik.vm.heapsize的值来获得Java Object Heap的最大值的,而ActivityManager类的成员函数getMemoryClass最终也通过读取这个系统属性的值来获得Java Object Heap的最大值。
这个Java Object Heap的最大值也就是我们平时所说的Android应用程序进程能够使用的最大内存。这里必须要注意的是,Android应用程序进程能够使用的最大内存指的是能够用来分配Java Object的堆。
在Android3.0以及更高的版本中,我们还可以在AndroidManifest.xml的application标签中增加一个值等于“true”的android:largeHeap属性来通知Dalvik虚拟机应用程序需要使用较大的Java Object Heap。事实上这个属性受限于手机内存,同时也会影响系统体验(毕竟系统总共可用的内存是固定的,一个应用程序用得多了,就意味意其它应用程序用得少了)。
Bitmap Memory也称为External Memory,它是用来处理图像的,这部分内存受Java Object Heap的大小限制的
Native Heap就是在Native Code中使用malloc等分配出来的内存,这部分内存是不受Java Object Heap的大小限制的,也就是它可以自由使用,当然它是会受到系统的限制。但是有一点需要注意的是,不要因为Native Heap可以自由使用就滥用,因为滥用Native Heap会导致系统可用内存急剧减少,从而引发系统采取激进的措施来Kill掉某些进程,用来补充可用内存,这样会影响系统体验。