用mat工具分析Android泄漏解决问题

    xiaoxiao2022-07-04  121

    内存泄漏是Android开发中比较难以解决的问题。这篇文章主要是来探讨如何使用mat工具进行内存泄漏的排查

    1.leakcanary上提示报错,发现有内存泄漏的提示

    2.打开Androidstudio 的Profile工具.菜单View-》ToolWindow-》Profiler (我本地的android studio版本是3.4),运行程序app

    3.在出现问题的页面来回切换

    4.点击Profiler的gc

    5.Dump java heap 

    6.在下方的Head Dump里面选择 app heap 然后 arrange by package 输入自己项目的包名

    发现SecondActivity的Allocations数量竟然等于7!说明存在SecondActivity的内存泄漏,使得SecondActivity的实例不被回收

    7.Export Heap Dump到指定目录,是一个后缀为.hprof格式的文件。

    然后从官网上下载MAT(一款eclipse下面的内存分析工具),具体网址:https://www.eclipse.org/mat/downloads.php

    mat工具下载好以后在mac环境下面报以下错误:

    不要怕,解决方案如下:

    https://blog.csdn.net/EaskShark/article/details/82664473

    8.终于能正确打开MAT工具了,导入刚才我们的dump 文件。这时候发现还是会报错,因为我们的hprof文件从AndroidStudio里面导出来以后还要进行格式上的转换。

    刚开始导入的时候会报错,这时候需要将dump进行转换,找到Androidsdk目录下platform-tools文件夹,里面有个hprof-conv工具,使用这个工具将dump文件转化为mat支持打开的格式。

    hprof-conv 1.hprof 2.hprof

    然后再用mat成功打开dump文件后,选择Histogram选项卡

    9.在上图的红框处输入过滤条件,这里我们可以选择我们发生内存泄漏的类名

     

    选择排除任何 软引用、虚引用等

    10.

    typesBySubscriber是EventBus的内部成员,这个可以EventBus里面的源码可以看到

    public class EventBus { /** Log tag, apps may override it. */ public static String TAG = "EventBus"; static volatile EventBus defaultInstance; private static final EventBusBuilder DEFAULT_BUILDER = new EventBusBuilder(); private static final Map<Class<?>, List<Class<?>>> eventTypesCache = new HashMap<>(); private final Map<Class<?>, CopyOnWriteArrayList<Subscription>> subscriptionsByEventType; private final Map<Object, List<Class<?>>> typesBySubscriber; private final Map<Class<?>, Object> stickyEvents;

    这个可以从EventBus的源码可以知道,typesBySubscriber是在构造器里面赋值,然后在register方法里面添加内容,在unregister方法里面清空内容。所以造成内存泄漏的原因就是EventBus的unregister方法没有正确调用。具体可以到代码里面排查下EventBus的unregister方法有没有正确调用即可。

     

     

     

     

    最新回复(0)