使用MAT (Memory Analyzer Tool)分析Andriod项目内存泄漏

前言:

在上一篇文章介绍了如何使用Android Monitor分析项目查找内存泄漏 ,本篇将介绍如何使用MAT(Memory Analyzer Tool)来分析和查找项目中内存泄漏的地方

MAT介绍:

MAT工具全称为Memory Analyzer Tool,一款详细分析Java堆内存的工具,该工具非常强大,为了使用该工具,我们需要hprof文件, Eclipse可以下载插件结合使用,也可以作为一个独立分析工具使用,下载地址:MAT

案例:(本篇还以上篇案例为例子)

public class CommUtil {
    private static CommUtil instance;
    private Context context;
    private CommUtil(Context context) {
        this.context=context;
    }
    public static CommUtil getInstance(Context context){
        if(null==instance){
            instance=new CommUtil(context);
        }
        return instance;
    }
}
public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        CommUtil instance = CommUtil.getInstance(this);
    }
}

(将Activity的实例被一个单例对象所持有,在旋转屏幕的时候造成内存泄漏)
将项目运行起来 打开Android Monitor观察Memory 并且试着多次旋转屏幕就会发现内存一直在增高


点击Dump java Heap生成一份内存快照hprof文件(若不了解如何生成hprof文件可以先去了解下如何使用Android Monitor分析项目查找内存泄漏)

Dump java Heap: 点击就会生成app运行内存快照.hprof文件

然后将APP完全退出 重新启动 打开Android Monitor 点击Dump java Heap点击生成一份还没操作前(旋转屏幕)的内存快照hprof文件(为以后作对比用)

现在就已经生成好了2份hprof文件 一份是没有旋转过屏幕的 一份是旋转过屏幕多次的 然后选中Android Studio 最左边的Captures 进行将hprof文件导出(因为MAT不支持Android Studio生成的未经转换的hprof文件 在导出过程中Android Studio会为我们转换好)



提示:导出的时候需要选择保存的目录以及文件名

终于MAT上场了 打开MAT 导入我们的2个hprof文件 Open File-->选择文件-->Leak Suspects Report-->Finish:



先来看看MAT都有哪些功能:

Overview视图
该视图会首页总结出当前这个Heap dump占用了多大的内存,其中涉及的类有多少,对象有多少,类加载器,如果有没有回收的对象,会有一个连接,可以直接参看(图中的Unreachable Objects Histogram)。


histogram视图: 列举内存中对象存在的个数和大小
Dominator tree视图:该视图会以占用总内存的百分比来列举所有实例对象,注意这个地方是对象而不是类了,这个视图是用来发现大内存对象的
Top Consumers: 该视图会显示可能的内存泄漏点
Duplicate Classes: 该视图显示重复的类等信息

本次重点围绕histogram视图讲解 其他3个视图也只是给我提供一些可能会造成内存泄漏的信息 点击打开Histograms视图


打开下面的面板Navigation History 选中histogram右键add to Compare Basket 添加到比较容器中(2个hprof文件都做以上同样的操作 将histogram 添加到比较容器中)

这时比较容器中就有2份hprof文件 一份未做任何操作 一份做了多次旋转屏幕(注意:导入的比较容器的顺序很重要 不然后面的数据看起来让人费解)然后点击Compare the results (红色感叹号)进行比较

通过比较后就会生成一个比较结果表ComPared Tables

但是这个表内容太多 如何快速过滤出我们与我们自己写的项目内容有关的呢 接着在Class Name 输入我们的项目包名

通上图就明显看出经过2个hprof比较 在我们操作了旋转多次屏幕后MainActivity的实例增加了然后我们就去分析下该hprof文件是什么导致MainActivity内存泄漏了

在Class Name处可以根据输入的类名去查找对应的对象实例(比如我们输入MainActivty):


Objects:实例个数
Shallow Heap:所占内存大小
Retained Heap:释放后能回收多少内存

通过上图看到了MainActivtiy在旋转屏幕后产生了2个实例 再去观察MainActivtiy具体被哪些对象引用呢 鼠标选中MainActivtiy 实例右键 选择with incoming references




看到MainActivty被引用的地方这么多 而且一屏还显示不完 我们又如何去判断是哪个导致内存泄漏的呢 MAT还有一个功能 就是通过遍历GC Root树去将那些有可能被GC回收的实例 将他们去除(备注:在GC Root树中能找到的对象绝对不存在有内存泄漏的实例 因为他们在运行时会被回收的嘛 只有找不到的那些才是)鼠标右键:

看那些单词就知道是什么意思 (排除 软引用 弱引用 虚引用) 最后我们就看到了Commutil的instance引用了MainActivty实例 而后旋转屏幕的时候 系统又创建了一个 所以有2个引用着MainActivtiy的实例 造成的内存泄漏:


然后就去修改代码:

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);

 CommUtil instance = CommUtil.getInstance(getApplicationContext());
  }
}

总结:

  1. 以上只是通过一个简单的案例教大家如何学会使用MAT去查找项目中泄漏的地方,在真实的项目开发过程中就不止有Commutil一个类 还有许多其他的类 需要一一去逐步分析是否有内存泄漏的可能。(体力活)
  2. 本案例是以旋转屏幕前和旋转屏幕后生成hprof文件通过MAT去分析内存是否泄漏,在实际案例中也是种套路去做,比如页面A跳转到页面B 在返回页面A内存却居高不下,然后就在跳转之前生成一份hprof文件 跳转之后再生成一份hprof文件 通过MAT一比较 检索出与页面A和B有关的内存情况,精确定位内存泄漏的地方,将范围缩小,省时省力。
  3. 对于检测内存泄漏的工具要灵活使用 哪种方便就用哪个。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容