Android开发中不免会遇到各类内存泄漏,若是不及时发现处理,会致使出现内存越用越大,可能会由于内存泄漏致使出现各类奇怪的crash,甚至可能出现因内存不足而致使APP崩溃。android
通常检测android内存泄漏都是使用android studio自带的Monitor工具结合mat,或者是使用第三方开源库工具:leakCanary.正则表达式
可是这二者都不是十分完美的方式:
Monitor工具结合mat有以下的问题:eclipse
不保存过去的数据,若是须要几个小时乃至几天的数据,这个工具就作不到;
有些内存泄露很慢,可能须要累计几天才有明显的增加,用自带的工具就不容易发现;
leakCanary有以下的问题:ide
工具比较死板,有些程序逻辑错误并不能检测出来.
另外有些内存能够优化的地方,使用leakcanary也是分析不出来的,好比程序中用到集合类,若是用完不调用clear并置为null的话,致使一个游离的对象集合会在内存中存在一段时间,只有等到下次GC检测到对象没有用的时候才会回收。
内存泄漏分析工具
Android的内存泄漏分析工具经常使用有Android Studio和基于eclipse的MAT(Memory Analyzer Tool)。经过二者配合,能够发挥出奇妙的效果。Android Studio可以快速定位内存泄漏的Activity,MAT能根据已知的Activity快速找出内存泄漏的根源。函数
第一步:强制GC,生成Java Heap文件
咱们都知道Java有一个很是强大的垃圾回收机制,会帮我回收无引用的对象,这些无引用的对象不在咱们内存泄漏分析的范畴,Android Studio有一个Android Monitors帮助咱们进行强制GC,获取Java Heap文件。工具
强制GC:点击Initate GC(1)按钮,建议点击后等待几秒后再次点击,尝试屡次,让GC更加充分。而后点击Dump Java Heap(2)按钮,而后等到一段时间,生成有点慢。post
生成的Java Heap文件会在新建窗口打开。测试
第二步:分析内存泄漏的Activity
点击Analyzer Tasks的Perform Analysis(1)按钮,而后等待几秒十几秒不等,便可找出内存泄漏的Activity(2)。优化
那么咱们就能够知道内存泄漏的Activity,由于这个例子比较简单,其实在(3)就已经能够看到问题所在,若是比较复杂的问题Android Studio并不够直观,不够MAT方便,若是Android Studio没法解决咱们的问题,就建议使用MAT来分析,因此下一步咱们就生成标准的hprof文件,经过MAT来找出泄漏的根源。this
第三步:转换成标准的hprof文件
刚才生成的Heap文件不是标准的Java Heap,因此MAT没法打开,咱们须要转换成标准的Java Heap文件,这个工具Android Studio就有提供,叫作Captures,右击选中的hprof,Export to standard .hprof选择保存的位置,便可生成一个标准的hprof文件。
第四步:MAT打开hprof文件
MAT的下载地址,使用方式和eclipse同样,这里就很少说了,打开刚才生成的hprof文件。点击(1)按钮打开Histogram。(2)这里是支持正则表达式,咱们直接输入Activity名称,点击enter键便可。
搜索到了目标的Activity
右击搜索出来的类名,选择Merge Shortest Paths to GC Roots的exclude all phantom/weak/soft etc. references,来到这一步,就能够看到内存泄漏的缘由,咱们就须要根据内存泄漏的信息集合咱们的代码去分析缘由。
第六步:根据内存泄漏信息和代码分析缘由
使用Handler案例分析,给出的信息是Thread和android.os.Message,这个Thread和Message配合一般是在Handler使用,结合代码,因此我猜想是Handler致使内存泄漏问题,查看代码,直接就在函数中定义了一个final的Handler用来定时任务,在Activity的onDestroy后,这个Handler还在不断地工做,致使Activity没法正常回收。
// 致使内存泄漏的代码protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_test);textView = (TextView) findViewById(R.id.text);final Handler handler = new Handler();handler.post(new Runnable() { @Override public void run() { textView.setText(String.valueOf(timer++)); handler.postDelayed(this, 1000); } });}
修改代码,避免内存泄漏
@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_test);textView = (TextView) findViewById(R.id.text);handler.post(new Runnable() { @Override public void run() { textView.setText(String.valueOf(timer++)); if (handler != null) { handler.postDelayed(this, 1000); } }});}private Handler handler = new Handler();@Overrideprotected void onDestroy() { super.onDestroy(); // 避免Handler致使内存泄漏 handler.removeCallbacksAndMessages(null); handler = null;}
从新测试,确保问题已经解决。