在项目中,有一个场景是某个 Activity 中会触发一个 BottomSheetDialogFragment
,并在某些情况下会常驻显示不消失。
然后就在线上遇到了一类 crash:
android.view.WindowManager$BadTokenException: Unable to add window -- token android.os.BinderProxy@c7a2c02 is not valid; is your activity running?
at android.view.ViewRootImpl.setView(ViewRootImpl.java:992)
at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:390)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:93)
at android.app.Dialog.show(Dialog.java:379)
at androidx.fragment.app.DialogFragment.onStart(DialogFragment.java:486)
目前推测的原因是:应用被重建(由应用启动运行时间很短推测,目测是应用退到了后台,被回收之后又被恢复)导致 Activity 被销毁重建,老的 DialogFragment 被恢复,但其绑定的还是被销毁前的 Activity Context(仅推测,还没有研究过源码),从而在恢复的时候被调用到 DialogFragment.onStart()
,最终出现了上述问题。
目前解决的方法是,在 DialogFragment
show 的时候绑定 tag,然后每次页面初始化的时候,先通过 FragmentManager#findFragmentByTag(tag)
来判断是否已经存在,如果存在则先移除老的。
fun removeOldObjIfExist(fragmentManager: FragmentManager) {
val old = fragmentManager.findFragmentByTag(SHOW_TAG) ?: return
fragmentManager.beginTransaction()
.remove(old)
.commitAllowingStateLoss()
}