使用软引用解决Handler内存泄露和显示Popupwindow、Dialog时提示"Unable to add Window-token is null"的问题

本文探讨如何使用软引用解决Handler可能导致的内存泄露问题,以及在Activity销毁后显示Popupwindow或Dialog时遇到的'Unable to add Window-token is null'异常。通过软引用,可以避免静态内部类持有非静态对象导致的内存泄漏,同时在显示Popupwindow或Dialog前检查Activity的状态,防止异常发生。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

通过软引用解决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);
    }
}
08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: FATAL EXCEPTION: main 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: Process: com.android.provision, PID: 2483 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running? 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.view.ViewRootImpl.setView(ViewRootImpl.java:2101) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:546) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:402) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:177) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.app.Dialog.show(Dialog.java:366) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.app.AlertDialog$Builder.show(AlertDialog.java:1131) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.widget.VideoView$5.onError(VideoView.java:615) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.media.MediaPlayer$EventHandler.handleMessage(MediaPlayer.java:3640) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.os.Handler.dispatchMessage(Handler.java:107) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.os.Looper.loopOnce(Looper.java:249) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.os.Looper.loop(Looper.java:337) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at android.app.ActivityThread.main(ActivityThread.java:9562) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at java.lang.reflect.Method.invoke(Native Method) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:648) 08-04 10:45:35.399 1000 2483 2483 E AndroidRuntime: at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1005) 08-04 10:45:35.402 1000 2483 2483 D OOMEventManagerFK: checkEventAndDumpForJE: 0
最新发布
08-05
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值