Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

时间:2022-11-12 17:03:22

之前碰到的OOM问题,终于很直白的呈现在我的眼前:我尝试了MAT,但是发现不怎么会用。直到今天终于发现了这个新工具:

当我们的App中存在内存泄露时会在通知栏弹出通知:

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

当点击该通知时,会跳转到具体的页面,展示出Leak的引用路径,如下图所示:

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

LeakCanary 可以用更加直白的方式将内存泄露展现在我们的面前。

以下是我找到的学习资料,写的非常棒:
1、LeakCanary: 让内存泄露无所遁形
2、LeakCanary 中文使用说明

AndroidStudio (官方)上使用LeakCanary 请移步:
https://github.com/square/leakcanary

Eclipse 上使用LeakCanary 请移步我的:
https://github.com/SOFTPOWER1991/LeakcanarySample-Eclipse

Android studio (自己弄的)上使用LeakCanary也可以看这个:

leakcanarySample_androidStudio

工程包括:

  1. LeakCanary库代码
  2. LeakCanaryDemo示例代码

使用步骤:

  1. 将LeakCanary import 入自己的工程

  2. 添加依赖:

    compile project(':leakcanary')

  3. 在Application中进行配置

    public class ExampleApplication extends Application {
    
      ......
    //在自己的Application中添加如下代码
    public static RefWatcher getRefWatcher(Context context) {
    ExampleApplication application = (ExampleApplication) context
    .getApplicationContext();
    return application.refWatcher;
    } //在自己的Application中添加如下代码
    private RefWatcher refWatcher; @Override
    public void onCreate() {
    super.onCreate();
    ......
    //在自己的Application中添加如下代码
    refWatcher = LeakCanary.install(this);
    ......
    } .....
    }
  4. 在Activity中进行配置

public class MainActivity extends AppCompatActivity {

    ......
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//在自己的应用初始Activity中加入如下两行代码
RefWatcher refWatcher = ExampleApplication.getRefWatcher(this);
refWatcher.watch(this); textView = (TextView) findViewById(R.id.tv);
textView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startAsyncTask();
}
}); } private void async() { startAsyncTask();
} private void startAsyncTask() {
// This async task is an anonymous class and therefore has a hidden reference to the outer
// class MainActivity. If the activity gets destroyed before the task finishes (e.g. rotation),
// the activity instance will leak.
new AsyncTask<Void, Void, Void>() {
@Override
protected Void doInBackground(Void... params) {
// Do some slow work in background
SystemClock.sleep(20000);
return null;
}
}.execute();
} }
  1. 在AndroidMainfest.xml 中进行配置,添加如下代码
        <service
android:name="com.squareup.leakcanary.internal.HeapAnalyzerService"
android:enabled="false"
android:process=":leakcanary" />
<service
android:name="com.squareup.leakcanary.DisplayLeakService"
android:enabled="false" /> <activity
android:name="com.squareup.leakcanary.internal.DisplayLeakActivity"
android:enabled="false"
android:icon="@drawable/__leak_canary_icon"
android:label="@string/__leak_canary_display_activity_label"
android:taskAffinity="com.squareup.leakcanary"
android:theme="@style/__LeakCanary.Base" >
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

5、测试结果

a、Toast显示(大概10秒左右显示)

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

b、通知显示

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

c、桌面自动添加的图表

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

d、内存泄露列表

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

e、内存泄露详细

Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

LogCat可以看到日志日下(hprof文件可以用MAT打开进行分析):

  1. 01-04 11:49:41.815 12967-13004/com.micky.leakcanarysamples I/dalvikvm: hprof: dumping heap strings to "/storage/emulated/0/Download/leakcanary/suspected_leak_heapdump.hprof".
  2. 01-04 11:49:42.020 12967-13004/com.micky.leakcanarysamples I/dalvikvm: hprof: heap dump completed (28850KB)

查看自动生成的AndroidManifest文件,LeakCanarySamples/app/build/intermediates/manifests/full/debug/AndroidManifest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.micky.leakcanarysamples"
  4. android:versionCode="1"
  5. android:versionName="1.0" >
  6. <uses-sdk
  7. android:minSdkVersion="10"
  8. android:targetSdkVersion="23" />
  9. <!-- To store the heap dumps and leak analysis results. -->
  10. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
  11. <android:uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
  12. <application
  13. android:name="com.micky.leakcanarysamples.BaseApplication"
  14. android:allowBackup="true"
  15. android:icon="@mipmap/ic_launcher"
  16. android:label="@string/app_name"
  17. android:supportsRtl="true"
  18. android:theme="@style/AppTheme" >
  19. <activity
  20. android:name="com.micky.leakcanarysamples.MainActivity"
  21. android:label="@string/app_name"
  22. android:theme="@style/AppTheme.NoActionBar" >
  23. <intent-filter>
  24. <action android:name="android.intent.action.MAIN" />
  25. <category android:name="android.intent.category.LAUNCHER" />
  26. </intent-filter>
  27. </activity>
  28. <activity android:name="com.micky.leakcanarysamples.TestActivity" />
  29. <service
  30. android:name="com.squareup.leakcanary.internal.HeapAnalyzerService"
  31. android:enabled="false"
  32. android:process=":leakcanary" />
  33. <service
  34. android:name="com.squareup.leakcanary.DisplayLeakService"
  35. android:enabled="false" />
  36. <activity
  37. android:name="com.squareup.leakcanary.internal.DisplayLeakActivity"
  38. android:enabled="false"
  39. android:icon="@drawable/__leak_canary_icon"
  40. android:label="@string/__leak_canary_display_activity_label"
  41. android:taskAffinity="com.squareup.leakcanary"
  42. android:theme="@style/__LeakCanary.Base" >
  43. <intent-filter>
  44. <action android:name="android.intent.action.MAIN" />
  45. <category android:name="android.intent.category.LAUNCHER" />
  46. </intent-filter>
  47. </activity>
  48. </application>
  49. </manifest>

如上所示LeakCanary给我们自动添加了两个Service和一个Activity,并添加了对SD卡的读写权限



It 's so simple.

注:

1、如果在Release模式下请使用RefWatcher.DISABLED

2、在Activity或Fragment 的 Destroy方法中添加检测(很好理解,就是判断一个Activity或Fragment想要被销毁的时候,是否还有其他对象持有其引用导致Activity或Fragment不能被回收,从而导致内存泄露)