Handler内存泄露的分析和解决办法以及实现延时执行操作的几种方法

一.Handler内存泄露的分析和解决办法

在进行异步操作时,我们经常会使用到Handler类。最常见的写法如下。

public class MainActivity extends Activity {  
    ......  
  
    private Handler mHandler = new Handler() {  
        @Override  
        public void handleMessage(Message msg) {  
            // your code  
        }  
    };  
}  

但是,这段代码很可能会引起内存泄漏。
当使用内部类或匿名内部类的方式创建Handler时,Handler对象会隐式地持有一个外部类对象的引用(这里的外部类是Activity)。
一般在一个耗时任务中会开启一个子线程,延迟执行操作,我们会使用到Handler对象。
但是,如果在任务未执行完时,Activity被关闭了,Activity已不再使用,此时由GC来回收掉Activity对象。
由于子线程未执行完毕,子线程持有Handler的引用,而Handler又持有Activity的引用,
这样直接导致Activity对象无法被GC回收,即出现内存泄漏。

为解决这个问题,下面这段代码中的Handler则是一个静态匿名内部类。
静态匿名内部类不会持有一个对外部类的隐式引用,因此Activity将不会被泄露。

如果你需要在Handler中调用外部Activity的方法,就让Handler持有一个对Activity的WeakReference,这样就不会泄露Activity的上下文了

public class MainActivity extends Activity {  
    ......  
  
    private final MyHandler mHandler = new MyHandler(this);  
  
    private static class MyHandler extends Handler {  
        private final WeakReference<MainActivity> mActivity;  
  
        public MyHandler(MainActivity activity) {  
            this.mActivity = new WeakReference<MainActivity>(activity);  
        }  
  
        @Override  
        public void handleMessage(Message msg) {  
            MainActivity mainActivity = mActivity.get();  
            if (mainActivity == null) {  
                return;  
            }  
            // your code here  
        }  
    }  
} 


最后我们尽量要在当前Activity的生命周期内,结束对所有回调函数和message的引用

   @Override
    protected void onDestroy() {
        super.onDestroy();
        mHandler .removeCallbacksAndMessages(null);
    }


二.实现延时执行操作的几种方法

开发过程中经常有各种各样的需求经常有各种各样的延迟操作需求

1.使用线程的休眠实现延时操作
				new Thread(new Runnable(){   
					   public void run(){   
					     try {
							Thread.sleep(2000);//设置休眠时间

							//your code 
						} catch (InterruptedException e) {
							e.printStackTrace();
						}   
					     
					   }   
					}).start();  
2.使用TimerTask实现延时操作
                        TimerTask task = new TimerTask() {
					@Override
					public void run() {
						//your code 
					}
				};
			   Timer timer = new Timer();
			   timer.schedule(task, 2000);//延迟2000ms执行task方法


3.使用Handler的postDelayed方法实现延时操作
				new Handler().postDelayed(new Runnable(){   
					  public void run() {   
						//your code 
					  }   
					 }, 2000); //延迟2000ms执行

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值