Android 非静态内部类导致内存泄漏

内存泄漏

一个不会被使用的对象,因为另一个正在使用的对象持有该对象的引用,导致它不能正常被回收,而停留在堆内存中。

内存泄漏的危害

最坏的情况,App可能会因为大量的内存泄漏而导致内存耗尽,引发Crash,如果内存未耗尽,App也会犹豫内存空间不足,出现频繁的GC(垃圾回收),每次一出GC都是非常耗时的阻塞性操作,会造成设备非常严重的卡顿,给用户的体验就是,手机无论做什么操作,都是卡的,这也是Android设备玩久了之后常见的现象。

泄漏代码(案例)

image.png

在sendGoodsMessage方法中,使用了一个非静态匿名内部类IMValueCallback,而这个非静态匿名内部类对其外部类存在一个隐式引用,其外部类在销毁之前,如果该非静态内部类的sendMessage异步任务还未完成,将会导致外部类的内存资源无法正常释放,造成了内存泄漏。

所以这个问题总结为: 非静态内部类中线程生命周期不可控,能否正常回收完全由线程的生命周期决定。如果线程是永久运行的,那么将永远无法释放,因为在Java中线程是垃圾回收机制的根源,在运行系统中DVM虚拟机总会硬件持有所有运行状态的进程的引用,结果导致处于运行状态的线程将永远不会被回收。

非静态内部类还有一种的情况的内存泄漏

非静态内部类中创建了一个静态实例,导致该实例的生命周期和应用ClassLoader级别,又因为该静态实例又会隐式持有其外部类的引用,所以导致其外部类无法正常释放,出现了泄漏问题。

深入分析

针对非静态内部类引发的内存泄漏这个点,进行深入分析

为什么非静态内部类对外部类会存在一个隐式引用? 为什么非静态内部类中存在异步任务,可能会导致其对应的外部类内存资源无法正常释放? 为什么非静态内部类中创建了一个静态实例,会导致内存泄漏?

深入剖析-隐式引用

image.png

为什么在非静态匿名内部类中,我们可以访问到外部类的testMethod方法?这就是隐式引用的作用。

通过编写一个简单的测试代码,进行深入的分析

源码和编译后的字节码">非静态匿名内部类的源码和编译后的字节码
image.png
image.png

args_size:这个代表着隐式引用的个数, 可以看出非静态匿名内部类中确实持有外部类的引用。

非静态内部类的源码和编译后的字节码
image.png
image.png
静态内部类的源码和编译后的字节码
image.png
image.png

结论:非静态内部类和非静态匿名内部类中确实都持有外部类的引用, 静态内部类中未持有外部类的引用
反编译线上版本, c.class 这个就是MessagePresender类的字节码

image.png

可以发现,在编译器编译过程中,帮我们隐式的传入了this这个参数,这也是为什么,我们平时在方法中能使用this这个关键字的原因。

了解了隐式引用,那么为什么它会是导致内存泄漏的根本原因? 这里又得说明一下,虚拟机的垃圾回收策略。

java垃圾回收机制

垃圾回收机制:Java采用根搜索算法,当GC Roots不可达时,并且对象finalize没有自救的情况下,才会回收。

回收对象:GC会收集那些不是GC roots且没有被GC roots引用的对象。

image.png

而这些引用就是对象之间的连线,垃圾回收的判定条件就在这些连线上,更详细的说明就不在这里描述,有兴趣的自行Google,或者查阅《深入理解Java虚拟机》。

解决方案

通过上述的分析,要预防非静态内部类的泄漏问题,就得管理好对象间的引用关系。

解决思路

去除隐式引用(通过静态内部类来去除隐式引用) 手动管理对象引用(修改静态内部类的构造方式,手动引入其外部类引用) 当内存不可用时,不执行不可控代码(Android可以结合智能指针,WeakReference包裹外部类实例)

image.png

最后

并不是所有的内部类只能使用静态内部类,只有在该内部类中的生命周期不可控制的情况下,我们要采用静态内部类,其它时候大家可以照旧。

如果了解了这些,比如Context、Handler、Timer、静态Acitivity、静态View、Thread等等造成的泄漏,也是能融会贯通的,只要大家多想一下它们之间的引用关系即可。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容

  • 一.Java内存分配结构复习 1.Java内存分配策略 上一篇Android内存管理分析总结中我们提到了Java内...
    Geeks_Liu阅读 776评论 5 7
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    _痞子阅读 1,586评论 0 8
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    apkcore阅读 1,193评论 2 7
  • 坚持读书打卡第四天。 提倡健康思维快乐人生,促使对方励志向上。
    冰雪连阅读 145评论 0 0
  • 一本书,一支钢笔,一张椅子,一张桌子,一瓶墨水,依然还是四年前的原班人马。人生就是这样,不论是处于何种境地...
    驴子自驾游阅读 839评论 8 10