Android OOM的那些事

本文详细探讨了Android进程的内存管理机制,包括虚拟内存与物理内存的映射、Dalvik虚拟机的堆内存限制及突破方法,以及如何通过命令行工具查看进程内存使用情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

基础

进程的内存空间是虚拟内存(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,从此不用担心看源码卡住
在这里插入图片描述
关于ThrowOutOfMemoryError方法的定义位于art/runtime/thread.cc
通过调用层次结构分析有以下7个地方,根据所在文件以及方法名,其实就可以知道,系统是如何抛出每一个OOM,这里就不继续细讲。
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值