DialogFragment 与 android.view.WindowManager$BadTokenException

这篇博客讨论了一个在Android应用中遇到的问题,即BottomSheetDialogFragment在特定情况下常驻显示导致的crash。问题源于应用被重建后,老的DialogFragment尝试绑定已销毁的ActivityContext。解决方法是在DialogFragment显示时检查并移除已存在的实例。通过FragmentManager的findFragmentByTag方法判断并移除旧对象,避免了WindowManager$BadTokenException的发生。

在项目中,有一个场景是某个 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()
}
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
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值