【Android内存泄漏检测】LeakCanary使用总结

时间:2023-01-28 10:05:14

一、什么是LeakCanary?

LeakCanary就是用来检测Android端内存泄漏的一个工具。能够检测Activity的泄漏

什么是内存泄漏?

Java 对象有时也会”长死不死“,GC 拿它没有办法,这种情况就是内存泄漏。造成这种情况的原因是:Java 对象被另一个生命周期更长对象持有,具有 可达性 ,这并不是我们想要的。

来自 <http://www.jianshu.com/p/3f1a1cc1e964>

内存泄漏的危害?

内存泄漏最终将导致内存溢出,也就是OOM,并且OOM发生在某一处,并不代表问题就是出在那里的,只是刚好有一块内存申请,此时内存又不够了才导致的,比较容易出现在申请比较大的内存的情况下。但是把这块申请的内存降低并不能从根本上解决问题。

这里有一个问题就是:内存的分配方式,分配位置,比如BitMap的分配,以及其他对象的分配?以及Activity对对象的持有,这个概念还不是很清晰。

github上的地址:https://github.com/square/leakcanary

一个原理的介绍内容在这里:

https://www.liaohuqiu.net/cn/posts/leak-canary/

https://www.liaohuqiu.net/cn/posts/leak-canary-read-me/

二、LeakCanary的工作原理是什么?

这三个都有讲解,并且还有直接通过源码来进行讲解的,后面可以针对这块,针对源码专门研究一下。

http://www.jianshu.com/p/a8900eb3de12

http://www.jianshu.com/p/5ee6b471970e

http://www.jianshu.com/p/3f1a1cc1e964

工作原理是:

LeakCanary会开启一个Service,这个Service能够在一个Activity被OnDestory之后,检测是否真的还有这个对象,如果还有,会尝试进行二次确认,如果这个对象还是没有被释放掉,则会获取当时的内存状态,生成一个heap的状态文件,然后分析文件,计算这个对象的GC ROOT的最短强引用路径,确定是否泄漏,如果有泄漏,Leak的APP就会有一条通知栏消息出现

检测流程图如下:

三、如何使用

1、在build.gradle中引入:

debugCompile 'com.squareup.leakcanary:leakcanary-android:1.3.1' // or 1.4-beta1
    releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3.1' // or 1.4-beta1
    testCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3.1' // or 1.4-beta1

或者,现在最新的1.5.1的也可以直接用:

dependencies{

compile'com.squareup.leakcanary:leakcanary-android:1.5.1'

}

2、同时,在Application的方法,在OnCreate中添加install

public class DemoApplication extends Application {
    @Override public void onCreate() {
        super.onCreate();
        LeakCanary.install(this);
    }
}

3、在添加之后,直接打出来应用的包,装上之后,会看到有一个附属应用出来,名字叫做Leaks

4、直接操作APP,如果发现有内存泄漏,则这个过程中会有通知栏提示,点击查看即可,可以将泄漏的信息分享,也可以将当时的heap dump的信息分享出来(但有可能这个heap dump不一定存在)

以检测出来的一个图为例:

LeakCanary的内存泄露提示一般会包含三个部分:

第一部分(LeakSingle类的sInstance变量)引用第二部分(LeakSingle类的mContext变量), 导致第三部分(MainActivity类的实例instance)泄露.

下方是一些更详细的原理的讲解:

1、不同的几种引用方式:强-软-弱-虚

2、LeakCanary这里,有一个原理的讲解叫做:

需要理解为什么要用WeakReference,因为GC的时候是不会对强引用的做处理,而软引用的是在内存不够用了才会被回收,而弱引用就是可以随时回收,所以需要用WeakReference对待检测的对象进行包裹引用,因为是弱引用的,因此在待检测对象调用了Destory或者Finish方法之后,被WeakReference引用的对象的生命周期结束,因为这个是弱引用,就会被GC检测到,这个时候GC会把该对象添加到ReferenceQueue中,如果GC结束该对象还是没有被加入到ReferenceQueue中,则说明可能存在内存泄漏,其实就是找到所有生命周期结束的对象,过滤掉会被回收的,剩下就是没有被回收的,然后会做二次GC,之后还是没有回收,就可以找到当时的内存状态,拿到不同对象的引用情况,得到一个hprof的文件,之后就能找到最短强引用的路径。

解决内存泄漏,通过以下方式:

然后我们需要解决:如何得到未被回收的对象。ReferenceQueue+WeakReference+手动调用 GC可实现这个需求。

  • WeakReference 创建时,传入一个 ReferenceQueue 对象。当被 WeakReference 引用的对象的生命周期结束,一旦被 GC 检查到,GC 将会把该对象添加到 ReferenceQueue 中,待ReferenceQueue处理。当 GC 过后对象一直不被加入 ReferenceQueue,它可能存在内存泄漏。

获得未被回收的 Object

  • 找到了未被回收的对象,如何确认是否真的内存泄漏?这里可以将问题转换为:未被回收的对象,是否被其他对象引用?找出其最短引用链。VMDebug + HAHA 完成需求。
    VM 会有堆内各个对象的引用情况,并能以hprof文件导出。HAHA 是一个由 square 开源的 Android 堆分析库,分析 hprof 文件生成Snapshot对象。Snapshot用以查询对象的最短引用链。

解析hprof

  • 找到最短引用链后,定位问题,排查代码将会事半功倍。

如下泳道图分析, LeakCanary 各个模块如何配合达到检测目的。

来自 <http://www.jianshu.com/p/3f1a1cc1e964>

【Android内存泄漏检测】LeakCanary使用总结的更多相关文章

  1. Android内存泄漏检测利器:LeakCanary

    Android内存泄漏检测利器:LeakCanary MAR 28TH, 2016 是什么? 一言以蔽之:LeakCanary是一个傻瓜化并且可视化的内存泄露分析工具 为什么需要LeakCanary? ...

  2. android 内存泄漏检测工具 LeakCanary 泄漏金丝雀

    韩梦飞沙 yue31313 韩亚飞 han_meng_fei_sha 313134555@qq.com 内存泄漏检测工具 android 内存泄漏检测工具 ======== 内存泄漏 就是  无用的对 ...

  3. Android 内存泄漏检测工具 LeakCanary(Kotlin版)的实现原理

    LeakCanary 是一个简单方便的内存泄漏检测框架,做 android 的同学基本都收到过 LeakCanary 检测出来的内存泄漏.目前 LeakCanary 最新版本为 2.7 版本,并且采用 ...

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

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

  5. Android内存泄漏的检测流程、捕捉以及分析

    https://blog.csdn.net/qq_20280683/article/details/77964208 Android内存泄漏的检测流程.捕捉以及分析 简述: 一个APP的性能,重度关乎 ...

  6. android 内存泄漏,以及检测方法

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

  7. 利用Android Studio、MAT对Android进行内存泄漏检测

    利用Android Studio.MAT对Android进行内存泄漏检测 Android开发中难免会遇到各种内存泄漏,如果不及时发现处理,会导致出现内存越用越大,可能会因为内存泄漏导致出现各种奇怪的c ...

  8. LeakCanary 来检查 Android 内存泄漏

    LeakCanary 来检查 Android 内存泄漏

  9. &lbrack;原理&rsqb; Android Native内存泄漏检测原理解析

    转载请注明出处:https://www.cnblogs.com/zzcperf/articles/11615655.html 上一篇文章列举了不同版本Android OS内存泄漏的检测操作(传送门), ...

随机推荐

  1. fibonacci数列(五种)

    自己没动脑子,大部分内容转自:http://www.jb51.net/article/37286.htm 斐波拉契数列,看起来好像谁都会写,不过它写的方式却有好多种,不管用不用的上,先留下来再说. 1 ...

  2. iOS中常用属性的关键字的使用说明

    属性关键字的作用 现在我们iOS开发中,基本都是使用ARC(自动引用计数)技术,来编写我们的代码.因此在属性property中我们经常使用的关键字有strong,weak,assign,copy,no ...

  3. SVN批处理更新文件夹

    SVN批处理更新文件夹 "C:\Program Files\TortoiseSVN\bin\TortoiseProc.exe" /command:update /path:&quo ...

  4. Firefox 32 支持 Public Key Pinning 对抗中间人攻击。

    Firefox 32 支持 Public Key Pinning 对抗中间人攻击.8月28日消息,即将发布的Firefox 32将支持Public Key Pinning机制,以防止中间人攻击.Pub ...

  5. 启动Activity的形式

    问:为什么service里面startActivity抛异常,activity不会? No1: 启动activity有两种形式: 1)直接调用Context类的startActivity方法:这种方式 ...

  6. 讯飞语音JavaWeb语音合成解决方案

    在线语音合成 将文字信息转化为声音信息,给应用配上"嘴巴".我们提供了众多极具特色的发音人(音库)供您选择.其合成音在音色.自然度等方面的表现均接近甚至超过了人声.这种语音合成体验 ...

  7. 增强for 可以用于ArrayList

    ArrayList<Integer> list=null; for(int i : list){ sum+=i; }

  8. 关于ehcache配置中timeToLiveSeconds和timeToIdleSeconds的区别

    在使用ehcache框架时,timeToLiveSeconds和timeToIdleSeconds这两个属性容易混淆,今天有空就记录一下,以防之后又忘记了. 首先来说明一下这两个属性分别有什么作用:( ...

  9. eclispe中使用python库 pyswip 进行prolog编程

    from pyswip import Prolog prolog = Prolog() prolog.assertz("father(michael,john)") prolog. ...

  10. gith命令行使用之上传和删除

    git这个工具的功能很强大,而使用git bash的命令行来进行git工具的操作尤为重要.而且我个人认为,用命令行进行git工具的操作比起图形界面的git工具,要更容易理解.图形界面的那个叫Torto ...