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()
}
2025-08-23 16:31:48.853 27906-27906 AndroidRuntime com.example.tapobulb E FATAL EXCEPTION: main Process: com.example.tapobulb, PID: 27906 java.lang.IllegalStateException: Fragment PresetFragment{f580327} (1290fafe-3f55-415e-9265-52d0afc5b47c) did not return a View from onCreateView() or this was called before onCreateView(). at androidx.fragment.app.Fragment.requireView(Fragment.java:1964) at com.example.tapobulb.PresetFragment.temperatureSelect_delegate$lambda$0(PresetFragment.kt:28) at com.example.tapobulb.PresetFragment.$r8$lambda$oFJfwoNazl7-VsWWQrHD6q4daXk(Unknown Source:0) at com.example.tapobulb.PresetFragment$$ExternalSyntheticLambda0.invoke(D8$$SyntheticClass:0) at kotlin.SynchronizedLazyImpl.getValue(LazyJVM.kt:74) at com.example.tapobulb.PresetFragment.getTemperatureSelect(PresetFragment.kt:27) at com.example.tapobulb.PresetFragment.onCreateView(PresetFragment.kt:52) at androidx.fragment.app.Fragment.performCreateView(Fragment.java:2963) at androidx.fragment.app.DialogFragment.performCreateView(DialogFragment.java:489) at androidx.fragment.app.FragmentStateManager.createView(FragmentStateManager.java:518) at androidx.fragment.app.FragmentStateManager.moveToExpectedState(FragmentStateManager.java:282) at androidx.fragment.app.FragmentManager.executeOpsTogether(FragmentManager.java:2189) at androidx.fragment.app.FragmentManager.removeRedundantOperationsAndExecute(FragmentManager.java:2100) at androidx.fragment.app.FragmentManager.execPendingActions(FragmentManager.java:2002) at androidx.fragment.app.FragmentManager$5.run(FragmentManager.java:524) at android.os.Handler.handleCallback(Handler.java:942) at android.os.Handler.dispatchMessage(Handler.java:99) at android.os.Looper.loopOnce(Looper.java:211) at android.os.Looper.loop(Looper.java:300) at android.app.ActivityThread.main(ActivityThread.java:8321) at java.lang.reflect.Method.invoke(Native Method) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:581) at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1028)报错了
最新发布
08-24
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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值