Android 优化Handler防止内存泄露

时间:2021-05-21

Android 优化Handler防止内存泄露

Demo描述:

Handler可能导致的内存泄露及其优化

1 关于常见的Handler的用法但是可能导致内存泄露

2 优化方式请参考BetterHandler和BetterRunnable的实现

package cc.cc; import java.lang.ref.WeakReference; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.app.Activity; /** * Demo描述: * Handler可能导致的内存泄露及其优化 * * 1 关于常见的Handler的用法但是可能导致内存泄露 * 请参考方法initHandler() * 2 优化方式请参考BetterHandler和BetterRunnable的实现 * * * */ public class MainActivity extends Activity { private Handler mHandler; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } /** * 常见的Handler的用法但是可能导致内存泄露 * * 比如在旋转屏幕时该Activity重新绘制. * 但是因为mHandler发送了一个延迟消息,所以消息队列持有mHandler对象 * 又由于new Runnable(){}持有外部类MainActivity的引用 * 所以Activity所占内存并不能向期望的那样被回收,这样就可能会造成内存泄漏. * * 这个例子中Handler的延迟时间比较久有20S,有点极端了,一般不会这么干; * 这里只是为了更好地说明这个问题就这么写代码了。 * */ private void initHandler() { mHandler = new Handler() { @Override public void handleMessage(Message msg) { super.handleMessage(msg); } }; // ......doing something // ......doing something // ......doing something // 发送延迟消息 mHandler.postDelayed(new Runnable() { @Override public void run() { } }, 1000 * 20); } /** * 以下为优化方式 * 1 在此处把BetterHandler和BetterRunnable都设计为静态类, * 这样它们就不会持有外部类的引用了. * 2 在BetterHandler中利用WeakReference持有Activity. * 常听说:"如果一个对象具有弱引用,那么当GC线程扫描的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存" * 其实准备地说应该是"如果一个对象只具有弱引用.........",即仅有弱引用而不存在对其的强引用才会将其回收. * 那么此处对Activity采用了弱引用,会不会导致该Activity被回收呢? * 答案是否定的。因为此处的Activity还在显示界面,当然存在其他对象对它的强引用。所以不会对其回收。 * * 经过这样的优化,当旋转屏幕时需要销毁原Activity时;消息队列持有Handler对象.但此时Handler对象不再持有Activity的引用. * 所以系统会回收该Activity所占内存.所以在handleMessage()中处理消息时需要判断Activity是否为空. * 比如此处20秒后才处理消息 这个时候Activity为空. */ private static class BetterHandler extends Handler{ private final WeakReference<Activity> activityWeakReference; public BetterHandler(Activity activity){ activityWeakReference=new WeakReference<Activity>(activity); } @Override public void handleMessage(Message msg) { super.handleMessage(msg); if (activityWeakReference.get()!=null) { //.....handle message } else { System.out.println("Activity==null"); } } } //同样采用静态内部类 private static class BetterRunnable implements Runnable{ @Override public void run() { // ......doing something } } //发送延迟消息 private void sendMessage(){ BetterHandler betterHandler=new BetterHandler(MainActivity.this); betterHandler.postDelayed(new BetterRunnable(), 1000 * 20); } }

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章