Android OOM的那些事

基础

进程的内存空间是虚拟内存(32位系统最大4G,64位系统4G*4G),然而程序的运行需要的是物理内存,虚拟内存和物理内存的映射关系,是通过内存管理单元(Memory Management Unt,MMU)实现的,主要是采取多级页表的方式。

Android 进程内存使用

android的内存空间,在32位系统的情况下。
Android OOM的那些事
Heap空间是需要我们重点关注的。其中C/C++申请的内存空间在Native Heap中,而Java申请的内存空间则在Dalvik Heap中。看起来我们的Heap可用有几个G,为什么还会OOM呢?

那是因为Dalvik虚拟机对VM Heapsize做了一个限制,可以通过命令 adb shell getprop查看。

其中dalvik.vm.heapgrowthlimit和dalvik.vm.heapsize都是单个进程可用的最大内存,当这2个值同时存在时,以dalvik.vm.heapgrowthlimit为准。假如我们要突破这个限制,需要在manifest中指定android:largeHeap为true,这样dvm heap最大可达heapsize。

通过命令adb shell cat /proc/$pid/status可以查看当前进程情况。其中VmPeak是虚拟内存最高峰值,VmSize是虚拟内存当前值。
Android OOM的那些事
还可以通过adb shell dumpsys meminfo $pid 查看。
Android OOM的那些事

会发现2个命令/proc/$pid/status和dumpsys meminfo的内存使用上数值会有差异。主要是dumpsys meminfo 在计算的时候会减去一些cache和buffer。

Bitmap内存使用的变迁,大家可以参考Android Bitmap变迁与原理解析(4.x-8.x)

OOM的种类

首先得点赞Android Code Search,从此不用担心看源码卡住
Android OOM的那些事
关于OOM的定义位于art/runtime/thread.cc
通过引用路径分析有一下7个地方,下面逐一看看。
Android OOM的那些事

未完待续