通过软引用解决Handler内存泄露的问题

通过软引用解决Handler内存泄露的问题

下面对软引用使用的方式适用于任何内部类,严格来说是通过软引用解决静态内部类无法调用当前类中的对象和方法的问题,真正解决内存泄露是需要将内部类改成静态内部类。
  当在一个类中按照如下方式创建一个Handler内部类时,使用Lint工具检测时会给出“This Handler class should be static or leaks might occur”的警告,原因是Handler内部类可能持有当前类的引用,导致即使该类不再被使用时系统仍无法回收这给类持有的对象,Android中内存泄露很多都是由于持有类中的对象时间太长导致,如果很多地方出现类似的代码会导致应用占用的内存不断上涨,最终导致程序崩溃。

private TextView mUserNameTxt = null;
class LoadDataHandler extends Handler{ 
    @Override 
    public void handleMessage(Message msg) {          
        super.handleMessage(msg); 
        switch(msg.what){ 
            case 0:{ 
                mUserNameTxt.setText(msg.obj.toString()); 
            }  
            break; 
            default:{
             } 
            break; 
        } 
    }
}

按照Lint的建议将Handler内部类改成static静态内部类后,由于不可能将当前类的所有全局对象都声明为static对象,所以会报“Cannot make a static reference to the non-static field”的错误,这时候可以使用软引用来解决这个问题,具体代码如下:

private TextView mUserNameTxt = null;static class LoadDataHandler extends Handler{ private SoftReference<MainActivity> activitySRF = null; public LoadDataHandler(MainActivity activity){ activitySRF = new SoftReference<MainActivity>(activity); } @Override public void handleMessage(Message msg) { super.handleMessage(msg); // 因为Handler是异步的,存在退出当前类之后才接收到handler消息的情况,并且软引用持有的对象会在堆内存不足时存在被回收的可能,所以这里需要判空处理 if(null == activitySRF || null == activitySRF.get()){ return; } switch(msg.what){ case 0:{ activitySRF.get().mUserNameTxt.setText(msg.obj.toString()); } break; default:{ } break; } }}

通过软引用解决销毁某个Activity后,仍然在Activity显示Popupwindow或者Dialog时提示”Unable to add Window-token is null”的问题
  在实际项目中踩过这个坑,特地分享出来,这个问题是由于在Activity中显示PopupWindow或者Dialog时,PopupWindow、Dialog还没显示出来就销毁了当前Activity导致(主要是快速切换)。因为PopupWindow和Dialog都是依附于当前Activity的某个View上的,当前Activity被销毁后依附的view为空,此时显示PopupWindow或者Dialog时会提示这个异常。
  这个问题也可以通过软引用来解决(通过判断软引用中的activity对象是否为空或者是否已经finish即可判断是否可以显示PopupWindow或者Dialog),具体代码如下:

private SoftReference<Activity> activitySRF = null;public void initDialog(Activity activity){ activitySRF = new SoftReference<Activity>(activity); if(null != activitySRF&&null!=activitySRF.get()&&!activitySRF.get().isFinishing()){ getWindow().requestFeature(Window.FEATURE_NO_TITLE); show(); setContentView(R.layout.hanzi_medal_dialoglayout); setCanceledOnTouchOutside(true); setCancelable(false); }}public void show(Activity activity, String medalName){ initDialog(activity); if(null != activitySRF&&null!=activitySRF.get()&&!activitySRF.get().isFinishing()){ TextView medalTxt = (TextView) findViewById(R.id.medalTxtId); if (!TextUtils.isEmpty(medalName)) { medalTxt.setText("恭喜您获得“" + medalName + "”勋章"); } new Handler(activitySRF.get().getMainLooper()).postDelayed(new Runnable() { @Override public void run() { if(null != activitySRF&&null!=activitySRF.get()&&!activitySRF.get().isFinishing()){ dismiss(); } } }, 3000); }}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,441评论 25 708
  • Android 内存泄漏总结 内存管理的目的就是让我们在开发中怎么有效的避免我们的应用出现内存泄漏的问题。内存泄漏...
    _痞子阅读 1,654评论 0 8
  • ###集合类泄漏 集合类如果仅仅有添加元素的方法,而没有相应的删除机制,导致内存被占用。如果这个集合类是全局性的变...
    RunningTeemo阅读 585评论 0 0
  • Android Studio JNI流程首先在java代码声明本地方法 用到native关键字 本地方法不用去实现...
    MigrationUK阅读 11,953评论 7 123
  • 鸟儿穿过湿淋淋的雨季,扑棱着潮湿的翅膀,划过枝头的西风和门前的柳塘,他们是要找寻温暖的窠巢还是梦的远方?树叶一片片...
    涵筠Han阅读 597评论 0 2