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

Posted 彬彬杨

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android7.1.1系统,Toast的Exception: android.view.WindowManager$BadTokenException解决相关的知识,希望对你有一定的参考价值。

最近在测谷歌支付的时候,7.1.1系统版本,我发现问题很多,比如 有个接口字段用了base64,但是我添加包的时候用的 java.util.base64 结果会崩溃报错,网上搜索后要用 android.util.base64 来解决,

今天在测谷歌支付的时候,发现突然有崩溃,看了下崩溃信息

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

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执行
        
    
 

以上是关于Android7.1.1系统,Toast的Exception: android.view.WindowManager$BadTokenException解决的主要内容,如果未能解决你的问题,请参考以下文章

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

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

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

Android7.1.1系统设置默认值大全

还必须保留哪些android 7.0 sdk组件才能支持android 7.1.1

AOSP android7.1.1-r16源码编译