背景

上周发现蘑菇街IM-Android代码里面。一些地方代码编写不当。存在内存泄漏的问题。在和疯紫交流的过程中。发现加深了一些理解,所以决定写一下分析思路,相互学习。

内存泄漏

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

内存泄漏的危害

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

泄漏代码(案例)

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

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

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

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

深入分析

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

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

深入剖析-隐式引用

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

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

非静态匿名内部类的源代码和编译后的字节码

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

非静态内部类的源代码和编译后的字节码

静态内部类的源代码和编译后的字节码

结论:非静态内部类和非静态匿名内部类中确实都持有外部类的引用。 静态内部类中未持有外部类的引用

那么这个引用是从何而来的?

能够通过蘑菇街线上代码进行反编译,我找了一个比較有代表性的代码,分析一下:

反编译线上版本号。 c.class 这个就是MessagePresender类的字节码

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

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

java垃圾回收机制

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

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

(图片来自网络:http://www.infoq.com/cn/articles/jvm-memory-collection)

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

解决方式

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

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

最后

并非全部的内部类仅仅能使用静态内部类。仅仅有在该内部类中的生命周期不可控制的情况下,我们要採用静态内部类,其他时候大家能够照旧。

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

Android 非静态内部类导致内存泄漏原因深入剖析的更多相关文章

  1. Android内存泄漏原因

    这段时间调试APP的时候,发现程序在加载了过多的bitmap后会崩溃.查看了日志,原来是发生了内存溢出(OOM).第一次遇到这样的问题,那就慢慢排查吧. 内存优化可以参考胡凯大神的博客Android内 ...

  2. android 常见内存泄漏原因及解决办法

    android常见内存泄漏主要有以下几类: 一.Handler 引起的内存泄漏. 在Android开发中,我们经常会使用Handler来控制主线程UI程序的界面变化,使用非常简单方便,但是稍不注意,很 ...

  3. Android 如何有效的解决内存泄漏的问题

    前言:最近在研究Handler的知识,其中涉及到一个问题,如何避免Handler带来的内存溢出问题.在网上找了很多资料,有很多都是互相抄的,没有实际的作用. 本文的内存泄漏检测工具是:LeakCana ...

  4. android 中如何分析内存泄漏

    转载:http://blog.csdn.net/fulinwsuafcie/article/details/8363218 前提条件: 1,电脑安装了java 运行环境 2,手机端开启了 USB 调试 ...

  5. Android开发 |常见的内存泄漏问题及解决办法

    在Android开发中,内存泄漏是比较常见的问题,有过一些Android编程经历的童鞋应该都遇到过,但为什么会出现内存泄漏呢?内存泄漏又有什么影响呢? 在Android程序开发中,当一个对象已经不需要 ...

  6. 老李分享:Android性能优化之内存泄漏1

    老李分享:Android性能优化之内存泄漏   前言 对于内存泄漏,我想大家在开发中肯定都遇到过,只不过内存泄漏对我们来说并不是可见的,因为它是在堆中活动,而要想检测程序中是否有内存泄漏的产生,通常我 ...

  7. Android 性能优化之内存泄漏检测以及内存优化(中)

    https://blog.csdn.net/self_study/article/details/66969064 上篇博客我们写到了 Java/Android 内存的分配以及相关 GC 的详细分析, ...

  8. Android中常见的内存泄漏

    为什么会产生内存泄漏? 当一个对象已经不需要再使用了,本该被回收时,而有另外一个正在使用的对象持有它的引用从而导致它不能被回收,这导致本该被回收的对象不能被回收而停留在堆内存中,这就产生了内存泄漏. ...

  9. Android开发——常见的内存泄漏以及解决方案(一)

    0. 前言   转载请注明出处:http://blog.csdn.net/seu_calvin/article/details/52333954 Android的内存泄漏是Android开发领域永恒的 ...

随机推荐

  1. Windows杂技

    WINDOWS下 ,在某目录下按住shift加鼠标右键,可以直接有当前目录的dos窗口 win10不能用 解决办法是在当前文件夹的地址栏输入cmd然后回车 发现Powershell可以当作dos执行相 ...

  2. 【Luogu】P1486郁闷的出纳员(Splay)

    题目链接 名副其实的调了一下午…… 每做一道题都是对我那不规范的Splay代码的刀刻斧凿一般的修正啊…… Splay.如果有一批员工不干了,那就找还能干的薪水最少的员工,把它splay到根,删除它的左 ...

  3. 如何在 Windows 上 使用 ONLYOFFICE 协作编辑文档

    ONLYOFFICE Document Server提供文档协作的服务功能,支持Word,Excel和PowerPoint的协作.但是这里告诉我们,需要进行文档管理和存储的二次开发. Please n ...

  4. Java众神之路(1)-语言介绍

    Java语言介绍 1.Java的历史 我个人认为,学习一种技术,不止要关注技术本身,也应该去了解一下它的发展史,这一方面是对技术本身的尊重,另一方面也是希望能够通过该技术的发展历史推测出其未来可能的发 ...

  5. iOS-Core Data基础

    Core Data基础 Core Data是一个API集合,被设计用来简化数据对象的持久存储. 在此先不普及概念,先通过一个简单的案例使用来感受一下Core Data的精妙之处. 在创建工程的时候勾选 ...

  6. 【bzoj4031】[HEOI2015]小Z的房间 Matrix-Tree定理+高斯消元

    [bzoj4031][HEOI2015]小Z的房间 2015年4月30日3,0302 Description 你突然有了一个大房子,房子里面有一些房间.事实上,你的房子可以看做是一个包含n*m个格子的 ...

  7. 42深入理解C指针之---指针与队列

    一.借助第40指针与链表的相关内容,稍微修改即可: 1.定义头文件queue.h代码如下: #include <stdlib.h> #include <stdio.h> #if ...

  8. Codeforces 835F Roads in the Kingdom (环套树 + DP)

    题目链接 Roads in the Kingdom 题意  给出一个环套树的结构,现在要删去这个结构中的一条边,满足所有点依然连通. 删边之后的这个结构是一棵树,求所有删边情况中树的直径的最小值. 显 ...

  9. va_list 简介

    原文:http://blog.sina.com.cn/s/blog_590be5290100qhxr.html va_list是一个宏,由va_start和va_end界定. typedef char ...

  10. CDOJ 879 摩天轮 dp+斜率优化

    原题链接:http://www.acm.uestc.edu.cn/#/problem/show/879 题意: 中文题 题解: 这是一道斜率dp的题. 先把$a$数组排个序. 令$dp[i][j]$表 ...