Android 优化Handler防止内存泄露
Android优化Handler防止内存泄露
Demo描述:
Handler可能导致的内存泄露及其优化
1关于常见的Handler的用法但是可能导致内存泄露
2优化方式请参考BetterHandler和BetterRunnable的实现
packagecc.cc;
importjava.lang.ref.WeakReference;
importandroid.os.Bundle;
importandroid.os.Handler;
importandroid.os.Message;
importandroid.app.Activity;
/**
*Demo描述:
*Handler可能导致的内存泄露及其优化
*
*1关于常见的Handler的用法但是可能导致内存泄露
*请参考方法initHandler()
*2优化方式请参考BetterHandler和BetterRunnable的实现
*
*
*
*/
publicclassMainActivityextendsActivity{
privateHandlermHandler;
@Override
protectedvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
}
/**
*常见的Handler的用法但是可能导致内存泄露
*
*比如在旋转屏幕时该Activity重新绘制.
*但是因为mHandler发送了一个延迟消息,所以消息队列持有mHandler对象
*又由于newRunnable(){}持有外部类MainActivity的引用
*所以Activity所占内存并不能向期望的那样被回收,这样就可能会造成内存泄漏.
*
*这个例子中Handler的延迟时间比较久有20S,有点极端了,一般不会这么干;
*这里只是为了更好地说明这个问题就这么写代码了。
*
*/
privatevoidinitHandler(){
mHandler=newHandler(){
@Override
publicvoidhandleMessage(Messagemsg){
super.handleMessage(msg);
}
};
//......doingsomething
//......doingsomething
//......doingsomething
//发送延迟消息
mHandler.postDelayed(newRunnable(){
@Override
publicvoidrun(){
}
},1000*20);
}
/**
*以下为优化方式
*1在此处把BetterHandler和BetterRunnable都设计为静态类,
*这样它们就不会持有外部类的引用了.
*2在BetterHandler中利用WeakReference持有Activity.
*常听说:"如果一个对象具有弱引用,那么当GC线程扫描的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存"
*其实准备地说应该是"如果一个对象只具有弱引用.........",即仅有弱引用而不存在对其的强引用才会将其回收.
*那么此处对Activity采用了弱引用,会不会导致该Activity被回收呢?
*答案是否定的。因为此处的Activity还在显示界面,当然存在其他对象对它的强引用。所以不会对其回收。
*
*经过这样的优化,当旋转屏幕时需要销毁原Activity时;消息队列持有Handler对象.但此时Handler对象不再持有Activity的引用.
*所以系统会回收该Activity所占内存.所以在handleMessage()中处理消息时需要判断Activity是否为空.
*比如此处20秒后才处理消息这个时候Activity为空.
*/
privatestaticclassBetterHandlerextendsHandler{
privatefinalWeakReferenceactivityWeakReference;
publicBetterHandler(Activityactivity){
activityWeakReference=newWeakReference(activity);
}
@Override
publicvoidhandleMessage(Messagemsg){
super.handleMessage(msg);
if(activityWeakReference.get()!=null){
//.....handlemessage
}else{
System.out.println("Activity==null");
}
}
}
//同样采用静态内部类
privatestaticclassBetterRunnableimplementsRunnable{
@Override
publicvoidrun(){
//......doingsomething
}
}
//发送延迟消息
privatevoidsendMessage(){
BetterHandlerbetterHandler=newBetterHandler(MainActivity.this);
betterHandler.postDelayed(newBetterRunnable(),1000*20);
}
}
如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!