Android OOM的那些事
基础
进程的内存空间是虚拟内存(32位系统最大4G,64位系统4G*4G),然而程序的运行需要的是物理内存,虚拟内存和物理内存的映射关系,是通过内存管理单元(Memory Management Unt,MMU)实现的,主要是采取多级页表的方式。
Android 进程内存使用
android的内存空间,在32位系统的情况下。
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是虚拟内存当前值。
还可以通过adb shell dumpsys meminfo $pid 查看。
会发现2个命令/proc/$pid/status和dumpsys meminfo的内存使用上数值会有差异。主要是dumpsys meminfo 在计算的时候会减去一些cache和buffer。
Bitmap内存使用的变迁,大家可以参考Android Bitmap变迁与原理解析(4.x-8.x)
OOM的种类
首先得点赞Android Code Search,从此不用担心看源码卡住
关于OOM的定义位于art/runtime/thread.cc
通过引用路径分析有一下7个地方,下面逐一看看。
未完待续