美文网首页
LeakCanary2.0版本原理简单查看

LeakCanary2.0版本原理简单查看

作者: 黑猫警长_01 | 来源:发表于2020-03-29 18:57 被阅读0次

这几天看了一下leakCanary2.0版本的源码,在这里做一下记录。
2.0版本使用kotlin重写的,使用起来也非常简单,省去了在Application中的注册,只需要在build.gradle文件中加入依赖

    debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.0'

就可以了,代码上什么都不用写,完全做到了无感知操作。

那么代码上什么都不用写,不用注册它怎么来进行调用,进行内存监测呢?
答案就在AppWatcherInstaller这个类上,它继承了ContentProvider,属于四大金刚,啊 不,四大组件之一。contentProvider的特点就是不用显示调用初始化,在执行完application的初始化后就会调用contentProvider的onCreate()方法。正是利用这一点,leakcanary把注册写在了这里面,有系统自动调用完成,对开发者完全无感知。
初次看leakCananry源码,我分了两部分来看的:1、添加要观察的对象 2、对对象进行观察
第一部分,添加观察对象
在AppWatcherInstaller的onCreate中调用了InternalAppWatcher的install方法
1、检查是否是在主线程(不是的话抛出异常)
2、添加生命周期的回调,在执行onDestory的时候调用objectWatcher的watch方法,观察这个activity的对象是否回收掉了

     ActivityDestroyWatcher.install(application, objectWatcher, configProvider)
//注册fragment的watcher
    FragmentDestroyWatcher.install(application, objectWatcher, configProvider)

//ActivityDestroyWatcher里的部分代码,再执行onDestory的时候开始进行对象检测

    private val lifecycleCallbacks =
       object : Application.ActivityLifecycleCallbacks by noOpDelegate() {
  override fun onActivityDestroyed(activity: Activity) {
    if (configProvider().watchActivities) {
      objectWatcher.watch(
          activity, "${activity::class.java.name} received Activity#onDestroy() callback"
      )
    }
  }
}

以上为注册流程。
第二部分就是检测对象的回收了
检测对象的回收 起始类是ObjectWatcher,顾名思义对象观察器。调用watch方法。ObjectWatcher.watch(watchedObject : Any,description : String)。watch方法里面做了如下操作:
在这个之前要先介绍一下,ObjectWatcher里面有两个集合分别是:
1) private val watchedObjects = mutableMapOf<String, KeyedWeakReference>() 这是一个map集合,用来存放要观察对象的key和弱引用,代码会为每个观察的对象生成一个唯一的key和弱应用
2)private val queue = ReferenceQueue<Any>()watchedObjects 这个队列和弱应用联合使用,当弱引用中的对象被回收后,这个弱引用会被放到这个队列中。换句话说就是只要存在这个队列中弱引用,就代表这个弱引用中所包含的对象被回收了。
下面开始watch方法里面的操作。
1、先移除watchedObjects 和queue 集合里面已经回收对象的弱引用。
2、通过uuid为当前观察的对象生成一个唯一的key,并把对象用弱应用包起来,放到watchedObjects 这个集合中,同时把queue 和弱应用关联起来。
3、checkRetainedExecutor.execute {moveToRetained(key) }这里是个延迟任务,延迟五秒后再次执行1操作,这个期间对象可能已经被回收了,所以需要再次移除一次
4、执行了3操作后,就可以通过watchedObjects 集合找到没有被回收的对象了。这时候就可以获取到没有被回收的对象的个数,大于0,进行一次gc操作(gc操作用的是Runtime.getRuntime() .gc(),源码上注释这个操作比system.gc()更可能触发gc操作)。

 var retainedReferenceCount = objectWatcher.retainedObjectCount
if (retainedReferenceCount > 0) {
  gcTrigger.runGc()
  retainedReferenceCount = objectWatcher.retainedObjectCount
}

5、再次获取未被回收的个数,这一步会有几个判断条件
1)如果个数小于5,不做操作等待5秒再次进行检查未回收的个数,一直循环,直到大于等于5个或者等于0个,为了防止频发回收堆造成卡顿。
2)大于5个后,如果处于debug模式,会再等20秒,再次执行4操作。防止debug模式会减慢回收
3)距离上次堆栈分析是否大于等于1分钟,如果没有超过一分钟,也需要再次延迟(1分钟-当前距离上次的时间)再次循环4操作

6、如果上面的条件都符合了,就可以开始进行堆栈的分析了
1)、获取到内容文件 Debug.dumpHprofData(heapDumpFile.absolutePath)
2)、objectWatcher.clearObjectsWatchedBefore(heapDumpUptimeMillis)该操作是去掉以前已经分析过的对象,也就是去除掉之前的没有回收掉的对象,不在本次分析范围内
3)、HeapAnalyzerService开启IntentService服务进行分析 具体分析就不写了,因为还没看懂
4)、把结果插入数据库(泄漏区分了应用本身的内存泄漏和类库的内存泄漏),并且发送通知

相关文章

  • LeakCanary2.0版本原理简单查看

    这几天看了一下leakCanary2.0版本的源码,在这里做一下记录。2.0版本使用kotlin重写的,使用起来也...

  • 不是很深入的分析LeakCanary2.0

    Leakcanary2.0 Leakcanary 使用 原理 Reference 、ReferenceQueue ...

  • leakcanary2.0原理分析

    1 问题描述 1.1 背景 Android内存泄漏检测工具--leakcanary,想必是众所周知。关于具体用法本...

  • 越狱防护(25)

    Tweak原理DYLD_INSERT_LIBRARIES 查看dyld源码(版本519.2.2) 当DYLD_IN...

  • mysql复习-实操

    简单信息查询 查看MYSQL版本: select version(); 查看当前用户: select user()...

  • Handler原理的简单版本

    Hanlder的存在主要是为了主线(UI)线程和子线程之间的通信(即Android的消息机制),原因是androi...

  • 13 LeakCanary源码解析

    【带着问题学】关于LeakCanary2.0你应该知道的知识点 前言 LeakCanary是一个简单方便的内存泄漏...

  • Linux查看系统信息

    查看系统版本 查看kernel内核: 查看版本Distribution(内置工具): 查看版本Distributi...

  • 查看版本

    1.查看服务器版本 redis-server -v Redis server v=4.0.11 sha=00000...

  • 版本查看

    cat /etc/issue输出: sudo lsb_release -a 将输出结果:

网友评论

      本文标题:LeakCanary2.0版本原理简单查看

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