android内存泄漏快速解决

明白原理

一般内存泄漏(traditional memory leak)的原因是:忘记释放分配的内存

逻辑内存泄漏(logical memory leak)的原因是:当应用不再需要这个对象,未释放该对象的所有引用。
一般内存泄漏如:Cursor忘记关闭等,此种场景存在一个标准的关闭方法,网络大把,不再展开,也比较好排查。实际操作中发现,Activity相关的逻辑泄漏占一大方向,泄漏的原因:
内部类,handler等,持有activity引用,导致Activity的需要释放的时候,释放不了。

这其中又以内部类表现最为突出:

  • 非静态内部类(默认)隐式持有外部类的的引用
  • 注意这和匿名不匿名没有关系
  • 内部类在干一些事情,外部类(如activity)在特殊环境下,比如内存不足,反复手动返回(monkey能跑出来),需要回收的时候,发现内部类还在持有他,从而产生无法回收的问题。就导致了泄漏。
tips: 在kotlin中 “嵌套类的类默认并不是内部类:他们并没有包含外部类的隐式引用”——《kotlin Action》

重点理解:第一条,第二条

一个内部类的示例代码:


内部类

这是安卓编码中经常会遇到的一种场景,即内部类访问外部类的private的成员变量或者方法。
这是可以的,为什么?java中private修饰的成员不是只能被成员所在类访问么,难道private真的失效了?
其实是编译器做了一些我们看不到的工作。反编译上面代码,我们来看看究竟发生了什么


用反编译工具处理后的代码

内部类MyRunnable里面多了一个MainActivity的成员变量,并且,在构造函数里面获得了外部类的引用
至此我们明白了:

为什么我们可以在非静态内部类中,直接调用外部类的成员?原因就是内部类,持有了外部类的一个引用 那就是this

内部类的调用外部类的成员的时候,其实是省略了 "this."的。

内部类和外部类之间,是通过一个外部类的引用进行调用的,而外部类的private属性通过编译器生成的我们看不见的静态方法,通过传入的外部类实例引用获取出来。通过还原,我们了解了非静态内部类跟外部类交互时的工作方式,以及非静态内部类为什么会持有外部类的引用。

辅助系列:

使用dump来查看是否泄漏

Android内存泄漏快速解决——三大工具
android内存泄漏快速解决——模板泄漏代码

解决系列:

android内存泄漏快速解决——正确处理handler

android内存泄漏快速解决——一个方法解决70%的泄漏
怎么解决这个问题,思路就是避免使用非静态内部类,定义内部类时,要么是放在单独的类文件中,要么就是使用静态内部类。因为静态的内部类不会持有外部类的引用,所以不会导致外部类实例的内存泄露。当你需要在静态内部类中调用外部的Activity时,我们可以使用弱引用来处理。
分享两个案例:leakCanary观测到的泄漏:

android内存泄漏快速解决——构造方法里不能出现非弱引用的对象的实例化

android内存泄漏快速解决——view意外销毁

android内存泄漏快速解决——忘记取消监听

其他正在处理中的泄漏:
android内存泄漏快速解决——fragment的rootview泄漏
android内存泄漏快速解决——系统源码泄漏
viewpager+fragment滑动泄漏

泄漏好文章整理:

https://juejin.im/entry/589542ed2f301e0069054007

https://www.jianshu.com/p/ac00e370f83d

你可能感兴趣的:(android内存泄漏快速解决)