码迷,mamicode.com
首页 > 其他好文 > 详细

LeakCanary:检测所有的内存泄露

时间:2015-05-11 01:25:25      阅读:219      评论:0      收藏:0      [点我收藏+]

标签:

        本文译自:https://corner.squareup.com/2015/05/leak-canary.html(LeakCanary是由Square公司刚刚开源用于查找Android内存泄露的库)

java.lang.OutOfMemoryError
        at android.graphics.Bitmap.nativeCreate(Bitmap.java:-2)
        at android.graphics.Bitmap.createBitmap(Bitmap.java:689)
        at com.squareup.ui.SignView.createSignatureBitmap(SignView.java:121)

          没有人喜欢OutOfMemory崩溃

在Square的注册界面,我们使用bitmap缓存绘制客户的签名,使用的bitmap大小等同于屏幕大小,当创建它时,我们有了大量的OOM崩溃。

技术分享
我们尝试了一些方法,但是没有一个可以解决问题:
  • 使用Bitmap.Config.ALPHA_8(签名不需要颜色)。
  • 捕获OutOfMemoryError,触发GC并多次重试(灵感来源于GCUtils)。
  • 我们没有想过从Java 堆分配位图,幸运的是,Fresco 也没有。(注:Fresco是Facebook用于图片缓存的开源库,可以关注一些)

我们曾看错了方向

bitmap的大小并不是问题,当内存将要用完的时候,OOM可以随处发生。它们更多的发生在你创建一个像Bitmap这种比较大的对象的情况下。OOM只是一个症状源自于更深层次的问题:内存泄露。

什么是内存溢出?

某些对象生命周期有限,当它们的工作完成以后,将会被回收。如果一个对象在其生命周期结束以后仍被内存中引用,将会导致内存泄露。当泄露积累过多,该应用将耗尽内存。

比如,在Activity.OnDestroy()方法被调用以后,这个activity的各种层级视图和它们关联的位图都应该被回收掉,如果一个后台运行的线程有该activity的引用,activity相对应的内存将不能被回收,这最终会导致OutOfMemoryError崩溃。

追踪内存溢出

追踪内存溢出是一个手动过程, Raizlabs 的  Wrangling Dalvik 对此有很好的描述。

下面是关键步骤:

  1. 通过 BugsnagCrashlytics,或者 Developer Console 了解内存溢出。
  2. 尝试重现该问题。你可能需要通过购买或者借用遇到崩溃问题的特定设备。(并非所有设备都会呈现这些泄漏) !你还需要弄清楚是哪一系列操作触发泄漏,有可能是蛮力。


  3. 当泄露发送时,记录堆栈内容(点击此处获取代码

  4. 使用MAT 或者YourKit 查看堆栈内容,并从中找出应该被回收的对象。

  5. 计算最短到 GC 根从该对象的强引用路径

  6. 找出哪个引用路径不应该存在,并修复内存溢出问题。

该库会在你遇到OOM的时候完成以上全部操作,这样你就可以专注于解决内存溢出问题了。


LeakCanary简介

LeakCanary是一个可以在你调试的时候检测内存泄露的Java开源库。

让我们看一个小例子:

class Cat {
}
class Box {
  Cat hiddenCat;
}
class Docker {
  static Box container;
}

// ...

Box box = new Box();
Cat schrodingerCat = new Cat();
box.hiddenCat = schrodingerCat;
Docker.container = box;

创建一个RefWatcher实例并设计一个监听对象
// We expect schrodingerCat to be gone soon (or not), let‘s watch it.
refWatcher.watch(schrodingerCat);

当检测到泄漏时,您自动获得不错的泄漏痕迹:
* GC ROOT static Docker.container
* references Box.hiddenCat
* leaks Cat instance

我们知道你忙着写功能,所以我们很容易设置。只是一个代码行,LeakCanary 将会自动检测活动泄漏:
public class ExampleApplication extends Application {
  @Override public void onCreate() {
    super.onCreate();
    LeakCanary.install(this);
  }
}

你会得到一个Notification和一个很好的展示界面:

技术分享


总结

在使用LeakCanary以后,我们在自己的应用中发现了许多内存泄露问题,我们甚至发现了一些Android SDK的内存泄露

结果是令人惊讶的,我们的OOM错误减少了94%。

技术分享

如果你想消灭OOM错误,现在就使用 LeakCanary 吧!

本文原创地址:http://blog.csdn.net/lilu_leo/article/details/45624735 转载请声明。

2015年,开始尝试玩一下微信公众账号:


技术分享





LeakCanary:检测所有的内存泄露

标签:

原文地址:http://blog.csdn.net/lilu_leo/article/details/45624735

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!