Android OOM问题的What Where How

知己知彼,百战不殆。
要想解决和避免OOM,必须先知道OOM是什么,在哪里会发生,最后才是怎样去解决OOM;

What?什么是OOM

OOM --> java.lang.OutOfMemoryError

内存溢出,应用内存占用过高,虚拟机无法再分配更多的内存,这时系统就会抛出OOM,JVM规范中划定了大部分区域的内存的管理,可参考JVM规范官方文档:https://docs.oracle.com/javase/specs/jls/se8/html/jls-15.html#d5e24307

Where & How? 哪里会出现OOM,怎么解决

首先要了解运行时内存区域的划分,可参考我的博客:
https://blog.csdn.net/u013360790/article/details/89510941
JVM除了程序计数器没有对内存做限制,其他内存区域都可以抛出OOM;
主要分为三大块,线程共享的堆和方法区,以及线程私有的栈,包括虚拟机栈和本地方法栈;


  • 堆中主要存放对象的实例,所以如果当堆无法存下要对象分配的空间,并且无法再进行扩展,这时就会抛出OOM;
    这个就比较好理解了,JDK1.8字符串也存储在堆中,几乎所有对象实例都在堆中,GC时间过长的也会引起OOM;
  • 方法区\MetaSpace
    方法区内存不足时也将会抛出OutOfMemoryError异常,而方法区主要存的类信息,以及常量,所以当我们系统本身内存不足时,进行加载大量的类或者常量时,也有可能出现OOM,在JDK1.7以前,连字符串常量都存储在方法区;

  • 虚拟机可以动态扩展,当扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常,比如在我们递归调用时,递归分为两步,一步递,一步归,所以在内存如果出现大量临时引用等,也有可能导致OOM;常见在Debug模式时,程序运行明显不如关闭Debug模式流畅,因为debug时需要额外存储大量调试信息;

根据JDK和Android 8.0源码,看下都什么地方Throw OutOfMemoryError

如果查找几个就会看出问题,系统都在什么情况下会抛出OOM,知道如何引起,自然知道如何避免和解决了;

  • 集合类
    大部分长度超出Integer.MAX_VALUE - 8 时,系统会抛出Required array size too large,Requested array size exceeds VM limit;
  • IO操作
    缓存过大,在Bitmap操作时,比如decodeStream(),可能抛出ImageUtils#decodeStream(InputStream, Rect, Options) threw an OOME,说明过大的文件加载到了内存中;
  • 线程泄漏
    创建线程超出JVM限制时会抛出unable to create new native thread,所以要注意线程的关闭,比如在使用ExecutorService记得shutDown;防止线程挂起或者阻塞或者等待获取锁时间过长,比如出现死锁;
  • 数据操作
    Cursor未进行及时close,在AbstractWindowedCursor中可以看到,Cursor存储了我们从数据库中查询回来的东西,所以在使用完要及时关闭,等待GC来回收是不靠谱的,因为有可能由于引用还在,所以无法回收;
  • 变量生命周期尽量短
    比如局部变量,因为引用存在栈中,所以方法弹栈后经过GC时就可以被回收,自然比成员变量生命周期短,及时的回收内存,自然可以有效避免内存溢出;
  • 引用泄漏
    这个非常好理解,例如泄漏Context,比如Activity这种Context往往是承载整个应用视图会引用大量变量,Context被生命周期更长的对象引用时,就无法回收Context了,就会出现内存泄漏,泄漏多了,自然溢出;
    举一反三,除了我们Android的Context,所有代码中的线程间交互,内部类对外部类的引用等都有可能在生命周期不一致情况下,造成内存泄漏,所以这种需要在编写代码时刻意防范;
  • 递归调用
    使用递归时注意出口,并且和For循环一样,尽量避免在内部循环创建对象,也要注意调用栈深度,否则也会报StackOverflowError;

还有很多细节平时多注意就好;
Android 可以使用LeakCanary来监控Context的泄漏,并能生成内存快照;
然后可以使用Android Studio或者Mat来对heep dump进行分析;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值