Android——内存管理基础

动态内存

《深入理解计算机系统》对动态内存分配器的定义是:维护着一个进程的虚拟存储器区域,成为”堆“,对于每一个进程,操作系统内核维护着一个变量brk(break)指向堆的顶部。

下面给出我关于动态内存分配方面的理解:

动态内存分配器,我的理解就是在程序执行的过程当中,动态的分配或释放虚拟内存。

有了动态内存分配的概念,自然就会有动态存储器的概念。所谓动态存储器,简而言之就是我们进程虚拟内存中的堆。堆在动态内存分配的过程中,会动态的增长或减少。堆在虚拟进程的结构当中是向上增长的,当堆内存不够时,还可以向操作系统请求额外的内存。

动态存储器分配器有两种基本分格:

  • 显式分配器:显示分配内存,显示释放内存。例如C和C++的malloc或者new分配内存,但是需要程序员显示调用free或者delete释放内存。
  • 隐式分配器:也叫做垃圾收集器,显示分配内存,自动释放未使用的已分配的块的过程叫做垃圾收集。例如:Java的垃圾回收机制。

垃圾收集器(garbage collector)

概念:自动的内存管理:显示地分配堆内存,隐式地释放堆内存。

功能:

  1. 分配堆内存
  2. 保证所有被引用的对象还在堆内存中
  3. 可以释放在运行的代码中不再引用的对象的堆内存

垃圾收集器避免了悬挂指针(dangling reference),原因是一个仍然被引用的对象永远不会内存回收并且也不会被认为已经被释放掉了。

垃圾收集器解决了空间泄漏(space leaks)问题,原因是它可以自动释放不再被引用的空间。

垃圾(garbage)

如果对象被引用着,那我们说它活着(live);如果对象不再被引用了,那我们说他死了(dead),术语称作垃圾(garbage)。

垃圾收集(garbage collection)

寻找并释放这些对象的空间的过程就做垃圾收集(garbage collection)。那么在什么时候会出发垃圾收集动作呢?一般来说整个堆或一部分被填满或者达到某一百分比数值时将被收集。

常见的三种垃圾收集器

串行(Serial):也被称为完全停顿(Stop-the-world)。当执行GC时,程序的线程都会被挂起。

并行(Parallel):圾收集工作被分成几部分,这些子部将会在不同的CPU上被同时执行。同时执行会使垃圾收集得更快,但是代价是会增加复杂性和潜在碎片。

并发(Concurrent):一个或者多个垃圾收集任务也可以并发的与应用程序同时执行。通常,一个并发收集器可以并发的执行垃圾收集的大部分工作,但是也会不可避免的引发一个小的停顿。

Android Garbage Collector发展

Android 1.0~2.2,Dalvik虚拟使用的垃圾收集机制有以下特点:

  1. Stop-the-word(完全停顿),也就是一个垃圾收集线程在执行的时候,其它的线程都停止;
  2. Full heap collection,也就是一次收集完全部的垃圾;
  3. 长等待,一次垃圾收集造成的程序中止时间通常都大于100ms。

Android 2.3~至今,Dalvik虚拟使用的垃圾收集机制得到了改进

  1. Cocurrent(并发),也就是大多数情况下,一个或多垃圾收集线程与其它线程是并发执行的;
  2. Partial collection,也就是一次可能只收集一部分垃圾;
  3. 短等待,一次垃圾收集造成的程序中止时间通常都小于5ms。

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表示垃圾收集造成的程序中止时间。

Android内存管理

Dalvik虚拟机的内存大体上可以分为Java Object Heap、Bitmap Memory和Native Heap三种。

Java Object 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

Bitmap Memory也称为External Memory,它是用来处理图像的,这部分内存受Java Object Heap的大小限制的

  • 在Android3.0之前,Bitmap Memory是在Native Heap中分配的,但是这部分内存同样计入Java Object Heap中,也就是说,Bitmap占用的内存和Java Object占用的内存加起来不能超过Java Object Heap的最大值。这就是为什么我们在调用BitmapFactory相关的接口来处理大图像时,会抛出一个OutOfMemoryError异常的原因。
  • 在Android3.0以及更高的版本中,Bitmap Memory就直接是在Java Object Heap中分配了,这样就可以直接接受GC的管理。

Native Heap

Native Heap就是在Native Code中使用malloc等分配出来的内存,这部分内存是不受Java Object Heap的大小限制的,也就是它可以自由使用,当然它是会受到系统的限制。但是有一点需要注意的是,不要因为Native Heap可以自由使用就滥用,因为滥用Native Heap会导致系统可用内存急剧减少,从而引发系统采取激进的措施来Kill掉某些进程,用来补充可用内存,这样会影响系统体验。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值