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

一、什么是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>

时间: 2024-08-17 20:59:43

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

Android内存泄漏检测-LeakCanary

square/leakcanary udacity/Sunshine-Version-2 添加LeakCanary依赖包 在主模块app下的build.gradle下添加如下依赖: debugCompile 'com.squareup.leakcanary:leakcanary-android:1.3.1' releaseCompile 'com.squareup.leakcanary:leakcanary-android-no-op:1.3.1' 开启LeakCanary 添加Applicat

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

韩梦飞沙 yue31313 韩亚飞 han_meng_fei_sha [email protected] 内存泄漏检测工具 android 内存泄漏检测工具 ======== 内存泄漏 就是  无用的对象没有被回收,占用着内存,使得可用内存变小了. 如何检测内存泄漏, 可以使用 LeakCanary来检测内存泄漏. leak  是 泄漏的意思.. Canary 是 金丝雀 的意思. 在运行 应用的时候, 泄漏金丝雀 如果检测到内存泄漏 会显示一个通知. ======== LeakCanary捕获

Android内存泄漏检测与MAT使用

内存泄漏基本概念 内存检测这部分,相关的知识有JVM虚拟机垃圾收集机制,类加载机制,内存模型等.编写没有内存泄漏的程序,对提高程序稳定性,提高用户体验具有重要的意义.因此,学习java利用java编写程序的时候,要特别注意内存泄漏相关的问题.虽然JVM提供了自动垃圾回收机制,但是还是有很多情况会导致内存泄漏. 内存泄漏主要原因就是一个生命周期长的对象,持有了一个生命周期短的对象的引用.这样,会导致短的对象在该回收时候无法被回收.Android中比较典型的有:1.静态变量持有Activity的co

android内存泄漏检测StrictMode和MAT工具使用

StrictMode说明 Android 2.3提供一个称为严苛模式(StrictMode)的调试特性,Google称该特性已经使数百个Android上的Google应用程序受益.那它都做什么呢?它将报告与线程及虚拟机相关的策略违例.一旦检测到策略违例(policy violation),你将获得警告,其包含了一个栈trace显示你的应用在何处发生违例.你可以强制用警告代替崩溃(crash),也可以仅将警告计入日志,让你的应用继续执行.策略的细节尚难确定,可以期待随Android的成熟Googl

Android内存泄漏查找和解决

Android内存泄漏查找和解决 目录: 内存泄漏的概念 一个内存泄漏的例子 Java中"失效"的private修饰符 回头看内存泄漏例子泄漏的重点 强引用与弱引用 解决内部类的内存泄漏 Context造成的泄漏 使用LeakCanary工具查找内存泄漏 总结 一.内存泄漏概念 1.什么是内存泄漏? 用动态存储分配函数动态开辟的空间,在使用完毕后未释放,结果导致一直占据该内存单元.直到程序结束.即所谓的内存泄漏. 其实说白了就是该内存空间使用完毕之后未回收 2.内存泄漏会导致的问题 内

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

https://blog.csdn.net/qq_20280683/article/details/77964208 简述: 一个APP的性能,重度关乎着用户体验,而关于性能检测的一个重要方面,就是内存泄漏,通常内存泄漏的隐藏性质比较强,不同于异常导致的程序Crash,在异常导致的Crash中,我们能够及时的发现程序问题的存在,并通过log日志定位到问题所在的具体位置,然后及时进行解决,而内存泄漏则不同,在APP中存在内存泄漏的情况下,用户在低频率短时间的使用中,并不能察觉到有什么异样,反之,随

基于Android Studio的内存泄漏检测与解决全攻略

自从Google在2013年发布了Android Studio后,Android Studio凭借着自己良好的内存优化,酷炫的UI主题,强大的自动补全提示以及Gradle的编译支持正逐步取代Eclipse,成为主流的Android开发IDE.Android Studio在为我们提供了良好的编码体验的同时,也提供了许多对App性能分析的工具,让开发者可以更方便分析App性能.Google在IO大会上一直告诫开发者不要无节制的使用手机内存,要注意一些不良的开发习惯会导致App的内存泄漏.虽然如今网上

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

在 Java 中,内存的分配是由程序完成的,而内存的释放则是由 Garbage Collecation(GC) 完成的,Java/Android 程序员不用像 C/C++ 程序员一样手动调用相关函数来管理内存的分配和释放,虽然方便了很多,但是这也就造成了内存泄漏的可能性,所以记录一下针对 Android 应用的内存泄漏的检测,处理和优化的相关内容,上篇主要会分析 Java/Android 的内存分配以及 GC 的详细分析,中篇会阐述 Android 内存泄漏的检测和内存泄漏的常见产生情景,下篇会

Android 性能篇 -- 带你领略Android内存泄漏的前世今生

基础了解 什么是内存泄漏? 内存泄漏是当程序不再使用到的内存时,释放内存失败而产生了无用的内存消耗.内存泄漏并不是指物理上的内存消失,这里的内存泄漏是指由程序分配的内存但是由于程序逻辑错误而导致程序失去了对该内存的控制,使得内存浪费. Java 内存分配策略 Java 程序运行时的内存分配策略有三种,分别是 静态分配 . 栈式分配 和 堆式分配 ,对应的三种存储策略使用的内存空间主要分别是 静态存储区(也称方法区) . 栈区 和 堆区 . ?? 静态存储区(方法区):主要存放 静态数据 . 全局