分析Handler引发的内存泄漏

作者: timmy_tan | 来源:发表于2019-05-16 12:11 被阅读7次

    分析工具

    android studio

    当前使用版本:android studio 3.4
    android studio 提供了Profiler工具可以提供可视化的内存观察,以及堆转储的功能。在实际使用过程中感觉太繁琐,因为我们没法确定是哪个页面发生了内存泄漏,所以就引用了LeakCanary

    2.LeakCanary

    当前使用版本:com.squareup.leakcanary:leakcanary-android:1.6.3
    引用官方的一句话"A memory leak detection library for Android",LeakCanary能帮我们动态监控内存泄漏,以及发生在哪个页面,同时还能自动的完成堆转储

    3.MAT

    当前使用版本:Eclipse Memory Analyzer Version 1.8.1
    引用官方的一句话"The Eclipse Memory Analyzer is a fast and feature-rich Java heap analyzer that helps you find memory leaks and reduce memory consumption.",MAT是一个功能强大的Java堆分析器

    内存泄漏代码

    • MainActivity
     @Override
      protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
          setContentView(R.layout.activity_main);
          findViewById(R.id.btn_no_static_class).setOnClickListener(v -> startActivity(new Intent(MainActivity.this, NoStaticClassActivity.class)));
      }
    
    • NoStaticClassActivity
        private Handler mHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
            }
        };
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_no_static_class);
            mHandler.postDelayed(new Runnable() {
                @Override
                public void run() {
                    Log.d(TAG, Thread.currentThread().getName());
                }
            }, 1000 * 60 * 10);
            finish();
        }
    

    分析

    • LeakCanary自动捕获到了信息:


      Screenshot_20190516-111448_LeakCanaryDemo.jpg

    这个Demo比较简单,其实通过LeakCanary就可以分析出问题出现在什么地方,当内容比较复杂时LeakCanary并没有那么直观的看出内存泄漏的原因,还是需要使用MAT工具

    • MAT分析.hprof文件
      LeakCanary会自动的将泄漏信息保存到手机的\Download\leakcanary-xxxx\目录下,将该目录下的.hprof文件复制到电脑,这个文件还不能直接用MAT工具打开,需要执行下面的一段命令,转换成能正常打开的文件
    hprof-conv.exe .\2019-05-16_11-13-28_699.hprof handle.hprof
    

    将转换好的handle.hprof文件使用MAT打开就会看见下面界面


    1.png

    然后点击用红色圈圈起来的Histigram,会看到下面的界面


    2.png
    然后在红色框中输入我们要分析的类名,在LeakCanary的界面中已经很明确的知道是NoStaticClassActivity这个类发生了泄漏,所以在框中输入NoS点击回车就会列来,右键点击我们搜索出来的类名 4.png

    点击后就会跳转到下面界面


    3.png
    这样我们就非常明确的看到了Thread持有了NoStaticClassActivity的应用,从中可以看到有MessageQueue,Message会很快想到和Handler有关,然后在分析具体代码。

    解决之后的代码

    • NoStaticClassActivity
     private final Handler mHandler = new MyHandle();
     private static class MyHandle extends Handler{
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
            }
        }
    

    在去点击按钮发现内存泄漏已经被我们修复了,本文重要的是说明工具的使用,代码有不严谨的地方请谅解

    相关文章

      网友评论

        本文标题:分析Handler引发的内存泄漏

        本文链接:https://www.haomeiwen.com/subject/brmbaqtx.html