本文发现了一类OOM(OutOfMemoryError),这类OOM的特点是崩溃时java堆内存和设备物理内存都充足,下文将带你探索并解释这类OOM抛出的原因。
关键词:
OutOfMemoryError, OOM,pthread_create failede,Could not allocate JNI Env
一、引子
对于每一个移动开发者,内存是都需要小心使用的资源,而线上出现的 OOM(OutOfMemoryError)都会让开发者抓狂,因为我们通常仰仗的直观的堆栈信息对于定位这种问题通常帮助不大。网上有很多资料教我们如何“紧衣缩食“的利用宝贵的堆内存(比如,使用小图片,bitmap 复用等),可是:
-
线上的 OOM 真的全是由于堆内存紧张导致的吗?
-
有没有 App 堆内存宽裕,设备物理内存也宽裕的情况下发生 OOM 的可能? 内存充裕的时候出现 OOM 崩溃? 看似不可思议,然而,最近笔者在调查一个问题的时候,通过自研的 APM 平台发现公司的一个产品的大部分 OOM 确实有这样的特征,即:
-
OOM 崩溃时,java 堆内存远远低于 Android 虚拟机设定的上限,并且物理内存充足,SD 卡空间充足
既然内存充足,这时候为什么会有 OOM 崩溃呢?
二、问题描述
在详细描述问题之前,先弄清楚一个问题:
什么导致了 OOM 的产生?
下面是几个关于 Android 官方声明内存限制阈值的 API:
图 2-1 Android 内存指标
通常认为 OOM 发生是由于 java 堆内存不够用了,即
图 2-2 Java 堆 OOM 产生原因
这种 OOM 可以非常方便的验证(比如: 通过 new byte[] 的方式尝试申请超过阈值maxMemory() 的堆内存),通常这种 OOM 的错误信息通常如下:
图 2-3 堆内存不够导致的 OOM 的错误信息
而前面已经提到了,本文中发现的 OOM 案例中堆内存充裕(Runtime.getRuntime().maxMemory() 大小的堆内存还剩余很大一部分),设备当前内存也很充裕(ActivityManager.MemoryInfo.availMem 还有很多)。这些 OOM 的错误信息大致有下面两种:
1 . 这种 OOM 在 Android6.0,Android7.0 上各个机型均有发生,文中简称为 OOM 一,错误信息如下:
图 2-4 OOM 一的错误信息
2 . 集中发生在 Android7.0 及以上的华为手机(EmotionUI_5.0 及以上)的 OOM,简称为 OOM 二,对应错误信息如下:
图 2-5 OOM 二的错误信息
三、问题分析及解决
3.1代码分析
Android 系统中,OutOfMemoryError 这个错误是怎么被系统抛出的?下面基于 Android6.0 的代码进行简单分析:
1. Android 虚拟机最终抛出OutOfMemoryError 的代码位于/art/runtime/thread.cc
图 3-1 ART Runtime 抛出的位置
2. 搜索代码可以发现以下几个地方调用了上述方法抛出 OutOfMemoryError 错误
3. 第一个地方是堆操作时