Android7.X系统,Toast的Exception: android.view.WindowManager$BadTokenException解决

这个BadTokenException是由谷歌系统产生的BUG,只有在系统7.x会产生,谷歌团队已经在8.0以上系统修复,但对于开发者来说面对android的7.x系统的用户不得不去解决这个BUG

BadTokenException报错日志如下:

Fatal Exception: android.view.WindowManager$BadTokenException: Unable to add window -- token android.os.BinderProxy@fcd9ef6 is not valid; is your activity running?
       at android.view.ViewRootImpl.setView(ViewRootImpl.java:806)
       at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:369)
       at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:94)
       at android.widget.Toast$TN.handleShow(Toast.java:459)
       at android.widget.Toast$TN$2.handleMessage(Toast.java:342)
       at android.os.Handler.dispatchMessage(Handler.java:102)
       at android.os.Looper.loop(Looper.java:186)
       at android.app.ActivityThread.main(ActivityThread.java:6491)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:914)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:804)

BadTokenException产生的原因及谷歌的解决方案

在Toast.show()之后,UI线程做了耗时的操作阻塞了Handlermessage的处理。7.X系统对TYPE_TOAST的Window类型做了超时限制,绑定了WindowToken,如果UI在这段时间内没有执行完,Toast.show()内部的handlermessage得不到执行,NotificationManageService那端会把这个Toast取消掉,同时把Toast对于的windowtoken置为无效。等App端真正需要显示Toast时,因为windowtoken已经失效,ViewRootImpl就抛出了上面的异常。

我们看一下谷歌源码是如何解决的,8.0以上系统Toast的handleShow 方法里面的一段代码,mWM.addView(mView, mParams),是被try catch 了,在7.x的系统是没有被catch住的(这里就不贴代码了)

看到谷歌的解决方案,所以我们7.x的解决方案其实也是把异常给catch 住。那如何catch住呢?我们看Toast的源码时里面有个mHandler的成员变量,我们想办法把分发消息的dispatchMessage(Message msg) 异常捕获不就可以了吗?这个其实和谷歌的解决方案一样。

 这里我们采用Java的反射机制

try {
      sField_TN = Toast.class.getDeclaredField("mTN");
      sField_TN.setAccessible(true);
      sField_TN_Handler = sField_TN.getType().getDeclaredField("mHandler");
      sField_TN_Handler.setAccessible(true);
      Object tn = sField_TN.get(toast);
      //获取原有的Handler
      Handler preHandler = (Handler) sField_TN_Handler.get(tn);
      //设置自定义的Handler
      sField_TN_Handler.set(tn, new SafelyHandlerWarpper(preHandler));
    } catch (Exception e) {
    }

  这里我们设置自定义的Handler

 /****
     * 自定义Handler catch处理异常
     */
    public static class SafelyHandlerWarpper extends Handler {
        private Handler impl;

        public SafelyHandlerWarpper(Handler impl) {
            this.impl = impl;
        }

        @Override
        public void dispatchMessage(Message msg) {
            try {
                super.dispatchMessage(msg);
            } catch (Exception e) {
            }
        }

        @Override
        public void handleMessage(Message msg) {
            impl.handleMessage(msg);//需要委托给原Handler执行
        }
    }

我们把dispatchMessage给catch住,handleMessage处理的消息具体实现还是原有的handler处理逻辑,不去影响原有的处理逻辑,我们只是catch住异常防止应用崩溃。

完整实现类如下:

/***
 * create by wuchu
 * 解决部分7.1.1手机崩溃Toast解决方案
 */
public class ZtAppCompatToast {

    private static Field sField_TN;
    private static Field sField_TN_Handler;

    private Toast mToast;


    static {
        //安卓7.0的做处理,其它版本系统的不用处理
        if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.N_MR1 && Build.VERSION.SDK_INT > Build.VERSION_CODES.M) {
            try {
                sField_TN = Toast.class.getDeclaredField("mTN");
                sField_TN.setAccessible(true);
                sField_TN_Handler = sField_TN.getType().getDeclaredField("mHandler");
                sField_TN_Handler.setAccessible(true);
            } catch (Exception e) {
            }
        }
    }

    public ZtAppCompatToast(Context context) {
        mToast = new Toast(context);
        setHook(mToast);
    }

    public Toast getToast() {
        return mToast;
    }

    public static Toast makeText(Context context,
                                 CharSequence text, int duration) {
        Toast toast = Toast.makeText(context, text, duration);
        setHook(toast);
        return toast;
    }

    public static Toast makeText(Context context,
                                 int textSrc, int duration) {
        Toast toast = Toast.makeText(context, textSrc, duration);
        setHook(toast);
        return toast;
    }

    private static void setHook(Toast toast) {
        //安卓7.0的做处理,其它版本系统的不用处理
        if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.N_MR1 && Build.VERSION.SDK_INT > Build.VERSION_CODES.M) {
            hook(toast);
        }
    }

    /*****
     *
     * @param toast
     */
    private static void hook(Toast toast) {
        try {
            Object tn = sField_TN.get(toast);
            Handler preHandler = (Handler) sField_TN_Handler.get(tn);
            sField_TN_Handler.set(tn, new SafelyHandlerWarpper(preHandler));
        } catch (Exception e) {
        }
    }

    /****
     * 自定义Handler catch处理异常
     */
    public static class SafelyHandlerWarpper extends Handler {
        private Handler impl;

        public SafelyHandlerWarpper(Handler impl) {
            this.impl = impl;
        }

        @Override
        public void dispatchMessage(Message msg) {
            try {
                super.dispatchMessage(msg);
            } catch (Exception e) {
            }
        }

        @Override
        public void handleMessage(Message msg) {
            impl.handleMessage(msg);//需要委托给原Handler执行
        }
    }

}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值