LeakCanary实现原理浅析

LeakCanary是一个在安卓平台上检测内存泄漏的工具库。

粗略的看了以下LeakCanary的实现原理。

LeakCanary地址

工程目录

工程目录
  • leakcanary-analyzer

    负责分析内存泄漏,主要使用了com.squareup.haha:haha库来分析

  • leakcanary-android

    负责android的接入

  • leakcanary-android-no-op

    空实现,就2个类,release后引用的空包

  • leakcanary-sample

    如何使用LeakCanary的示例

  • leackcanary-watcher
    负责监视对象是否泄漏

工作流程

  1. 安装LeakCanary
    安装LeakCanary过程中注册监听Activity的生命周期。
  2. 监听Activity生命周期,当Activity发生destroyed的时候,弱引用Activity为KeyedWeakReference。
  3. 当主线程空闲的时候执行GC操作,判断弱引用是否释放。
  4. 弱引用没有释放,则找到内存泄漏,进行内存泄漏分析,之后通知和展示。

源码解析

看源码的时候,从初始化入手,然后找到核心链路。

  1. 安装过程

初始化的安装流程最终调用的是

   /**
   *
   * @param application
   * @param listenerServiceClass 默认传递 DisplayLeakService.class
   * @param excludedRefs 排除的情况 默认为AndroidExcludedRefs.createAppDefaults().build() 
   * @return
   */
  public static RefWatcher install(Application application,
      Class<? extends AbstractAnalysisResultService> listenerServiceClass,
      ExcludedRefs excludedRefs) {
    //是否在分析的进程(HeapAnalyzerService进程)
    if (isInAnalyzerProcess(application)) {
      return RefWatcher.DISABLED;
    }
    //在桌面显示内存泄漏Activity(DisplayLeakActivity)的图标
    enableDisplayLeakActivity(application);
    //启用分析的回调 结果会启用HeapAnalyzerService进行HeapDump分析来找出泄漏的源头
    HeapDump.Listener heapDumpListener =
        new ServiceHeapDumpListener(application, listenerServiceClass);
    //监视器 leakcanary核心部分 后面会分析
    RefWatcher refWatcher = androidWatcher(application, heapDumpListener, excludedRefs);
    //把Activity列为监视器的监视对象 通过监听Activity发生destroyed
    ActivityRefWatcher.installOnIcsPlus(application, refWatcher);
    return refWatcher;
  }

install主要做了3件事情

1. 在桌面启用DisplayLeakActivity的图标
2. 初始化监听器RefWatcher,并监听Activity
3. 在监听到有内存泄漏后调用heapDumpListener来启用HeapAnalyzerService
  1. RefWatcher
    RefWatcher是leackcanary的核心,他负责监听内存泄漏是否发生。

RefWatcher的成员变量

  //监听执行器 实现类 AndroidWatchExecutor 核心代码 Looper.myQueue().addIdleHandler(IdleHandler)
  private final Executor watchExecutor;
  //负责日志输出 实现类 AndroidDebuggerControl 通过Debug.isDebuggerConnected()来判断是否输出日志
  private final DebuggerControl debuggerControl;
  //GC触发器 抄AOSP代码 https://android.googlesource.com/platform/libcore/+/master/support/src/test/java/libcore/java/lang/ref/FinalizationTester.java
  private final GcTrigger gcTrigger;
  //进行headDump操作 实现类 AndroidHeapDumper 核心代码 Debug.dumpHprofData(heapDumpFile.getAbsolutePath()); 另外还做了一个5s超时处理 超时实现方法可以参考下^_^
  private final HeapDumper heapDumper;
  //保存在监听的对象 如果GC后还存在里面 说明内存泄漏了
  private final Set<String> retainedKeys;
  //内存被成功回收会进入该队列 然后会更新retainedKeys
  private final ReferenceQueue<Object> queue;
  //在install的时候传入的ServiceHeapDumpListener 负责dump后的回调
  private final HeapDump.Listener heapdumpListener;
  //排除项
  private final ExcludedRefs excludedRefs;

此处需要一个图来解释RefWatcher工作流程

  1. 泄漏分析

找到泄漏点后开始启用HeapAnalyzerService进行泄漏分析。

//获取泄漏分析结果 核心代码 ShortestPathFinder.findPath(Snapshot snapshot, Instance leakingRef) 
AnalysisResult result = heapAnalyzer.checkForLeak(heapDump.heapDumpFile, heapDump.referenceKey);
//交给DisplayLeakService进行展示处理
AbstractAnalysisResultService.sendResultToListener(this, listenerClassName, heapDump, result);

找到内存泄漏的路径的核心代码
大概思路是从GCRoot出发,广度优先搜索到leakingRef就返回,其中利用excludedRefs进行剪枝。

Result findPath(Snapshot snapshot, Instance leakingRef) {
    clearState();
    canIgnoreStrings = !isString(leakingRef);
    //搜索队列里增加GCRoot
    enqueueGcRoots(snapshot);

    boolean excludingKnownLeaks = false;
    LeakNode leakingNode = null;
    //优先找toVisitQueue队列中的 找完再找toVisitIfNoPathQueue,而路径中包含toVisitIfNoPathQueue里的元素则标示excludingKnownLeaks为true
    while (!toVisitQueue.isEmpty() || !toVisitIfNoPathQueue.isEmpty()) {
      LeakNode node;
      if (!toVisitQueue.isEmpty()) {
        node = toVisitQueue.poll();
      } else {
        node = toVisitIfNoPathQueue.poll();
        if (node.exclusion == null) {
          throw new IllegalStateException("Expected node to have an exclusion " + node);
        }
        excludingKnownLeaks = true;
      }

      // 找到泄漏点 跳出循环
      if (node.instance == leakingRef) {
        leakingNode = node;
        break;
      }
      //判断是否搜索过了 看了代码 按我的理解 这里没必要搞toVisitSet,toVisitIfNoPathSet,visitedSet 保留visitedSet就够了
      if (checkSeen(node)) {
        continue;
      }
      
      if (node.instance instanceof RootObj) {
        visitRootObj(node);
      } else if (node.instance instanceof ClassObj) {
        visitClassObj(node);
      } else if (node.instance instanceof ClassInstance) {
        visitClassInstance(node);
      } else if (node.instance instanceof ArrayInstance) {
        visitArrayInstance(node);
      } else {
        throw new IllegalStateException("Unexpected type for " + node.instance);
      }
    }
    return new Result(leakingNode, excludingKnownLeaks);
  }
  1. 内存泄漏通知和展示

在拿到泄漏路径后,交给DisplayLeakService进行处理。代码很简单就发了个通知。

 @Override protected final void onHeapAnalyzed(HeapDump heapDump, AnalysisResult result) {
    String leakInfo = leakInfo(this, heapDump, result, true);
    CanaryLog.d(leakInfo);

    boolean resultSaved = false;
    boolean shouldSaveResult = result.leakFound || result.failure != null;
    if (shouldSaveResult) {
      heapDump = renameHeapdump(heapDump);
      resultSaved = saveResult(heapDump, result);
    }

    PendingIntent pendingIntent;
    String contentTitle;
    String contentText;

    if (!shouldSaveResult) {
      contentTitle = getString(R.string.leak_canary_no_leak_title);
      contentText = getString(R.string.leak_canary_no_leak_text);
      pendingIntent = null;
    } else if (resultSaved) {
      pendingIntent = DisplayLeakActivity.createPendingIntent(this, heapDump.referenceKey);

      if (result.failure == null) {
        String size = formatShortFileSize(this, result.retainedHeapSize);
        String className = classSimpleName(result.className);
        if (result.excludedLeak) {
          contentTitle = getString(R.string.leak_canary_leak_excluded, className, size);
        } else {
          contentTitle = getString(R.string.leak_canary_class_has_leaked, className, size);
        }
      } else {
        contentTitle = getString(R.string.leak_canary_analysis_failed);
      }
      contentText = getString(R.string.leak_canary_notification_message);
    } else {
      contentTitle = getString(R.string.leak_canary_could_not_save_title);
      contentText = getString(R.string.leak_canary_could_not_save_text);
      pendingIntent = null;
    }
    showNotification(this, contentTitle, contentText, pendingIntent);
    afterDefaultHandling(heapDump, result, leakInfo);
  }

DisplayLeakActivity就不分析了,主要负责内存泄漏的展示。

总结

本文只是粗略的梳理LeakCanary流程,其中还有许多细节没有提及。

本文分析的是master分支上的代码,只支持监听Activity泄漏,不过了解了整个流程后,我们可以加入更多的监听对象,如WebView Fragment等。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342