[crash 分析] android.view.WindowManager$BadTokenException: Unable to add window

【背景介绍】

  最近项目版本更新,上线后的版本出现了这样一例crash,具体的堆栈信息如下:

android.view.WindowManager$BadTokenException: Unable to add window -- token android.os.BinderProxy@c5e3e83 is not valid; is your activity running?
                      at android.view.ViewRootImpl.setView(ViewRootImpl.java:584)
                      at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:282)
                      at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:90)
                      at android.app.Dialog.show(Dialog.java:298)
                      at android.support.v7.app.AlertDialog$Builder.show(AlertDialog.java:918)
                      at com.baidu.testalertdialogcrash.MainActivity$1$1.run(MainActivity.java:26)
                      at android.os.Handler.handleCallback(Handler.java:739)
                      at android.os.Handler.dispatchMessage(Handler.java:95)
                      at android.os.Looper.loop(Looper.java:135)
                      at android.app.ActivityThread.main(ActivityThread.java:5386)
                      at java.lang.reflect.Method.invoke(Native Method)
                      at java.lang.reflect.Method.invoke(Method.java:372)
                      at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:927)
                      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:722)

【问题分析】

  出现上述crash,从信息提示里可以看出:在AlertDialog的show前,对应的View已经被销毁,所以想在销毁的view上展示AlertDialog肯定会发生此类的crash.为了重现这个问题,编写了如下测试代码:


public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button btn = (Button) findViewById(R.id.test_crash);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                finish();

                new Handler().postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        new AlertDialog.Builder(MainActivity.this).show();
                    }
                }, 2000);
            }
        });
    }
}

  通过测试发现,点击测试按钮后,销毁了对应的activity,此时延迟2s展示的alertDialog发生crash,完全验证了我们的猜测。知道了问题的原因,解决方法就十分简单了,在展示dialog前,判断对应的view是否已经被销毁即可。具体做法如下:

new Handler().postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        // 在这里增加逻辑保护,如果view已经被销毁,就不再显示对应的AlertDialog,避免发生crash
                        if (MainActivity.this.isFinishing()) {
                            return;
                        }
                        new AlertDialog.Builder(MainActivity.this).show();
                    }
                }, 2000);

【个人总结】

  1. AlertDialog 创建依赖于View的context,在展示这个AlertDialog前,需要增加view是否被销毁的判断处理

【原创声明】

  转载请注明出处,个人Blog主页:个人博客地址
  Blog地址:http://blog.csdn.net/csdn_lexli/article

你可能感兴趣的:(crash分析)