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

[java]  view plain copy
  1. package cc.cc;  
  2.   
  3. import java.lang.ref.WeakReference;  
  4. import android.os.Bundle;  
  5. import android.os.Handler;  
  6. import android.os.Message;  
  7. import android.app.Activity;  
  8. /** 
  9.  * Demo描述: 
  10.  * Handler可能导致的内存泄露及其优化 
  11.  *  
  12.  * 1 关于常见的Handler的用法但是可能导致内存泄露 
  13.  *   请参考方法initHandler() 
  14.  * 2 优化方式请参考BetterHandler和BetterRunnable的实现 
  15.  *    
  16.  * 关于Handler可以参考我以前写的: 
  17.  * http://blog.csdn.net/lfdfhl/article/details/40016165 
  18.  *  
  19.  * 参考资料: 
  20.  * 1 http://blog.csdn.net/krislight/article/details/9391403 
  21.  * 2 http://blog.csdn.net/myarrow/article/details/14223493 
  22.  * 3 http://2dxgujun.com/post/2014/09/11/Handler-Leaks-Solution.html 
  23.  * 4 http://blog.csdn.net/FeeLang/article/details/39059705 
  24.  *   Thank you very much 
  25.  * 
  26.  */  
  27. public class MainActivity extends Activity {  
  28.     private Handler mHandler;  
  29.     @Override  
  30.     protected void onCreate(Bundle savedInstanceState) {  
  31.         super.onCreate(savedInstanceState);  
  32.         setContentView(R.layout.main);  
  33.     }  
  34.       
  35.       
  36.     /** 
  37.      * 常见的Handler的用法但是可能导致内存泄露 
  38.      *  
  39.      * 比如在旋转屏幕时该Activity重新绘制. 
  40.      * 但是因为mHandler发送了一个延迟消息,所以消息队列持有mHandler对象 
  41.      * 又由于new Runnable(){}持有外部类MainActivity的引用 
  42.      * 所以Activity所占内存并不能向期望的那样被回收,这样就可能会造成内存泄漏. 
  43.      *  
  44.      * 这个例子中Handler的延迟时间比较久有20S,有点极端了,一般不会这么干; 
  45.      * 这里只是为了更好地说明这个问题就这么写代码了。 
  46.      *  
  47.      */  
  48.     private void initHandler() {  
  49.         mHandler = new Handler() {  
  50.             @Override  
  51.             public void handleMessage(Message msg) {  
  52.                 super.handleMessage(msg);  
  53.             }  
  54.         };  
  55.   
  56.         // ......doing something  
  57.         // ......doing something  
  58.         // ......doing something  
  59.   
  60.         // 发送延迟消息  
  61.         mHandler.postDelayed(new Runnable() {  
  62.             @Override  
  63.             public void run() {  
  64.   
  65.             }  
  66.         }, 1000 * 20);  
  67.     }  
  68.       
  69.       
  70.       
  71.     /** 
  72.      * 以下为优化方式 
  73.      * 1 在此处把BetterHandler和BetterRunnable都设计为静态类, 
  74.      *  这样它们就不会持有外部类的引用了. 
  75.      * 2 在BetterHandler中利用WeakReference持有Activity. 
  76.      *  常听说:"如果一个对象具有弱引用,那么当GC线程扫描的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存" 
  77.      *  其实准备地说应该是"如果一个对象只具有弱引用.........",即仅有弱引用而不存在对其的强引用才会将其回收. 
  78.      *  那么此处对Activity采用了弱引用,会不会导致该Activity被回收呢? 
  79.      *  答案是否定的。因为此处的Activity还在显示界面,当然存在其他对象对它的强引用。所以不会对其回收。 
  80.      *   
  81.      * 经过这样的优化,当旋转屏幕时需要销毁原Activity时;消息队列持有Handler对象.但此时Handler对象不再持有Activity的引用. 
  82.      * 所以系统会回收该Activity所占内存.所以在handleMessage()中处理消息时需要判断Activity是否为空. 
  83.      * 比如此处20秒后才处理消息 这个时候Activity为空. 
  84.      */  
  85.     private static class BetterHandler extends Handler{  
  86.         private final WeakReference<Activity> activityWeakReference;  
  87.         public BetterHandler(Activity activity){  
  88.             activityWeakReference=new WeakReference<Activity>(activity);  
  89.         }  
  90.         @Override  
  91.         public void handleMessage(Message msg) {  
  92.             super.handleMessage(msg);  
  93.             if (activityWeakReference.get()!=null) {  
  94.                 //.....handle message  
  95.             } else {  
  96.                 System.out.println("Activity==null");  
  97.             }  
  98.         }  
  99.     }  
  100.       
  101.     //同样采用静态内部类  
  102.     private static class BetterRunnable implements Runnable{  
  103.         @Override  
  104.         public void run() {  
  105.             // ......doing something  
  106.         }  
  107.           
  108.     }  
  109.       
  110.     //发送延迟消息  
  111.     private void sendMessage(){  
  112.         BetterHandler betterHandler=new BetterHandler(MainActivity.this);  
  113.         betterHandler.postDelayed(new BetterRunnable(), 1000 * 20);  
  114.     }  
  115.       
  116.   
  117. }  

你可能感兴趣的:(Handler可能导致的内存泄露及其优化)