android应用中有时会出现OutOfMemory Error报错(也就是OOM crash),这种报错的一般会有两种原因:
- 内存溢出:一次性分配内存过大,超过了heapsize限制
- 内存泄露:应用存在内存泄露,累积超过了heapsize限制
第一种,我们先不讨论。主要讨论第二种--内存泄露。
什么是内存泄露
java的内存泄露,是指有些对象不能及时释放,导致所在进程的所占的内存不断增大,并且不能够被GC回收掉,随着时间的延长,这个进程的内存会越来越大,极限情况就是发生OOM crash。
java GC的机制
GC回收就是把没有引用的对象会释放掉。当一个对象还有被其他对象引用的时候,这个对象是释放不了的。如果这个对象被多次创建,且都被其他对象引用,就会导致内存不断增大。
Android平台内存泄露注意点
1、cursor没有关闭
如下在使用Cursor后,没有关闭,Cursor导致这个也会导致内存泄露
try {
Cursor c = queryCursor();
....
c.close();
} catch (Exception e) {
}
在try语句中如果发生异常,就会导致c.close()没有调用到,这时就发生了异常,正确写法如下
Cursor c;
try {
c = queryCursor();
} catch (Exception e) {
} finally{
if (c != null) {
c.close();
}
}
2、HandlerThread使用错误
如下代码就是HandlerThread的错误用法
public class MyActivity extends Activity {
@Override
public void onCreate(BundlesavedInstanceState) {
super.onCreate(savedInstanceState);
...
Thread thread = new HandlerThread("leak", Process.THREAD_PRIORITY_BACKGROUND);
thread.start();
MyHandler mHandler = new MyHandler(thread.getLooper());
}
@Override
public void onDestroy() {
super.onDestroy();
}
}
我们可以看到HandlerThread thread初始化,并且调用start()后,就一直就没有其他操作了。
如
Android内存泄漏分析及调试分析,HandlerThread在start()后,就会调用run()函数,而run()是死循环,这样thread就得不到释放,进而MyActivity也得不到释放。
正确的写法该是这样的
public class MyActivity extends Activity {
private Thread mThread;
@Override
public void onCreate(BundlesavedInstanceState) {
super.onCreate(savedInstanceState);
...
mThread = new HandlerThread("leak", Process.THREAD_PRIORITY_BACKGROUND);
mThread.start();
MyHandler mHandler = new MyHandler(mThread.getLooper());
}
@Override
public void onDestroy() {
super.onDestroy();
mThread.quit(); //或者mThread.quitSafely();
}
}