LeakCanary:检测所有的内存泄漏

移动开发
在Square的注册过程中,我们在bitmap上 绘制了一个用户的签名。这个bitmap和设备的屏幕大小相当,在创建它的时候,我遇到了相当数量的OOM导致的崩溃。这样的问题怎么下手处理掉!

 原文: LeakCanary: Detect all memory leaks! 

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

没人喜欢OutOfMemoryError

在Square的注册过程中,我们在bitmap上 绘制了一个用户的签名。这个bitmap和设备的屏幕大小相当,在创建它的时候,我遇到了相当数量的OOM导致的崩溃。

signature.png

我们试过了几种方法,没有一个解决了我们的问题:

  • 使用Bitmap.Config.ALPHA_8(签名是不需要颜色的)

  • 捕获OutOfMemoryError,触发垃圾回收然后重试几次(从GCUtils 获得的启发)

  • 我们没有考虑过将bitmap分配在堆内存之外,那个时候Fresco 还没出现。

我们看待问题的方式是不对的

bitmap的大小本身不是什么问题。当内存快要满了的时候,OOM随时随地都可能发生。尤其是在创建大对象的时候更容易发生,比如bitmap。OOM一般代表着更深层次的问题:内存泄漏。

什么是内存泄漏?

有些对象只有有限的生命周期。当它们的任务完成之后,它们将被垃圾回收。如果在对象的生命周期本该结束的时候,这个对象还被一系列的引用,这就会导致内存泄漏。随着泄漏的累积,app将消耗完内存。

比如,在Activity.onDestroy()被调用之后,view树以及相关的bitmap都应该被垃圾回收。如果一个正在运行的后台线程继续持有这个Activity的引用,那么相关的内存将不会被回收,这最终将导致OutOfMemoryError崩溃。

寻找内存泄漏

寻找内存泄漏是一个人工操作的过程,在Raizlabs的 Wrangling Dalvik 系列中描述得很清楚。

下面是关键的步骤:

  1. 通过 BugsnagCrashlytics, 或者 Developer Console了解OOM

  2. 主动重现问题。你可能需要买或者借或者偷一个遭遇了崩溃的特殊设备(并不是所有的设备上都会发生内存泄漏!)。你还需要找出是什么串在一起导致了内存泄漏。

  3. 当OOM出现时进行堆转储(dump the heap)(这里介绍了如何做).

  4. 使用MAT或YourKit内存检测工具检测内存的变化,并找出哪个对象应该被垃圾回收;

  5. 从那个对象到GC roots推断最短的强引用路径;

  6. 在路径中找出不存在的引用,并修复memory leak;

要是有一个库可以为你做完所有的事情,让你专注于修复内存泄漏的问题,那该有多好。

LeakCanary介绍

LeakCanary 是一个开源的在debug版本中检测内存泄漏的java库。

让我们来看看一个cait的例子:

  1. class Cat { 
  2. class Box { 
  3.   Cat hiddenCat; 
  4. class Docker { 
  5.   static Box container; 
  6.   
  7. // ... 
  8.   
  9. Box box = new Box(); 
  10. Cat schrodingerCat = new Cat(); 
  11. box.hiddenCat = schrodingerCat; 
  12. Docker.container = box; 

创建一个RefWatcher实例,然后给它一个对象让它观察:

 
  1. // We expect schrodingerCat to be gone soon (or not), let's watch it. 
  2. refWatcher.watch(schrodingerCat); 

当检测出泄漏的时候,你会自动得到一个漂亮的泄漏线索:

  1. * GC ROOT static Docker.container 
  2. * references Box.hiddenCat 
  3. * leaks Cat instance 

我们知道你的时间宝贵,因此我们让它非常好设置。只需几行代码,LeakCanary就能自动检测Activity的泄漏:

  1. public class ExampleApplication extends Application { 
  2.   @Override public void onCreate() { 
  3.     super.onCreate(); 
  4.     LeakCanary.install(this); 
  5.   } 

 

当内存不足时,会有一个通知和良好的展示界面:

leaktrace.png

结论

在启用LeakCanary之后,我们发现和修复了许多内存泄漏的问题。我们甚至发现了一些 Android SDK中的泄漏

结果是非常令人惊奇的,我们现在减少了94%的oom崩溃问题。

oom_rate.png

如果你想终结OOM崩溃,现在就安装LeakCanary!

责任编辑:倪明 来源: Square engineering blog
相关推荐

2023-10-31 16:40:38

LeakCanary内存泄漏

2017-05-23 21:00:47

Android开源项目内存泄漏

2018-12-07 10:52:08

内存泄漏方法

2010-09-25 11:07:45

Java内存泄漏

2011-06-16 09:28:02

C++内存泄漏

2009-06-16 11:20:22

内存泄漏

2010-09-26 15:38:33

JVM内存泄漏

2022-09-28 10:35:31

JavaScript代码内存泄漏

2015-04-17 10:35:51

c++c++程序内存泄漏检测代码

2021-03-04 17:21:49

内存检测泄漏

2024-03-11 08:22:40

Java内存泄漏

2023-12-18 10:45:23

内存泄漏计算机服务器

2012-02-22 21:28:58

内存泄漏

2015-03-30 11:18:50

内存管理Android

2021-08-05 15:28:22

JS内存泄漏

2021-08-09 09:54:37

内存泄漏JS 阿里云

2019-01-30 18:24:14

Java内存泄漏编程语言

2020-06-08 09:18:59

JavaScript开发技术

2017-11-09 16:07:00

Web应用内存

2009-07-17 16:19:16

Swing核心组件
点赞
收藏

51CTO技术栈公众号