内存泄漏的原因
一直以来以为只有C/C++才存在内存泄漏的问题,没想到拥有内存回收机制的Java也可能出现内存泄漏。C/C++存在指针的概念,程序中需要使用指针变量时,就从内存中开辟一块区域,并把该区域的首地址赋值给一个指针,这样程序才可操作该指针指向的内存区域。因为C/C++设计上的原因,手工分配的内存,也要手工来释放,如malloc/free是C中分配/释放内存的运算符,而new/delete则是C++中新增的分配/释放内存的运算符。Java设计之初就是能够自动回收内存,可是有些时候因为某些因素,内存回收机制并不会都奏效。情况之一是调用了非java接口,比如调用了jni接口,jni中C/C++的内存就要手工回收;情况之二是调用了外部服务,使用完毕就得手工通知外部服务去回收;情况之三是异步处理,实时的内存回收显然顾不上异步处理的任务。
内存泄漏的场景
在Android开发中,内存泄漏可能发生在如下几个场景:1、查询操作后,没有关闭游标Cursor;
2、刷新适配器Adapter时,没有重用convertView对象;
3、Bitmap对象使用完毕,没有调用recycle方法回收内存;
4、给系统服务注册了监听器,却没有及时注销;
5、Activity引用了耗时对象,造成页面关闭时无法释放被引用的对象;
内存泄漏的发现
检查app是否发生内存泄漏,有三个办法:1、在代码中定期检查当前进程占用的内存大小。
2、使用ADT自带DDMS插件的heap工具,去发现是否有内存溢出。
如果在Heap的Tab中发现提示“DDMS Heap updates are NOT ENABLED for this client”,则在菜单“Preferences”——“Android”——“DDMS”中打开“Thread updates enabled by default”。如果还不行,则在DDMS的devices窗口中,选择调试的进程,点击上方的堆栈图标(Update Heap)。
3、通过内存分析工具MAT(Memory Analyzer Tool,一个Eclipse插件),找到内存泄露的对象。devices窗口上方堆栈图标右侧有个向下箭头的图标(DUMP HPROF file),这是heap工具生成的app内存统计文件,MAT读取该文件后会给出方便阅读的信息,配合它的查找、对比功能,就可以定位内存泄漏的原因。
注意MAT依赖于插件BIRT Chart Engine,得先安装这个BCE插件,然后才能安装MAT插件。
内存泄漏的预防
关闭游标
游标Cursor不光用于SQLite数据库,也可用于ContentProvider的ContentResolver对象,以及DownloadManager查询下载任务,相关介绍参见《 Android开发笔记(三十一)SQLite游标及其数据结构》。预防游标产生的内存泄漏,可在每次