从Dialog闪退看WMS部分源码2020-12-09

一 问题引出

前两天做一个需求,需求中有界面有加载网络,加载网络的时候要弹个dialog显示正在加载,等到服务器返回数据后,dialog消失。
开发前,搜索了一下,发现google推荐DialogFragment,但是有很多评论说坑比较多,所以我就没有用它而是直接用了Dialog,然后就出问题了。(其实想想也知道,google推出DialogFragment取代Dialog肯定是Dialog不够好的,就像用DataStore取代SharedPreferences一样。)
出现的具体问题就是,activity 生命周期内去show一个dialog,然后activity 在destroy之后再去dismiss这个dialog,虽然在dismiss之前有判断dialog.isShowing(),但还是会闪退。因为activity destroy之后,WindowManagerGlobal中的mView中已经清空了。再次去移除时,找不到,直接throw new IllegalArgumentException("View=" + view + " not attached to window manager");

val dialog:Dialog = Dialog(activity, themeId)
if (!activity.isDestroyed) {
    dialog.show()
}
...
if (dialog != null && dialog.isShowing) {
    dialog.dismiss()  //闪退
}

二 报错日志:

11-30 17:26:08.599 E/AndroidRuntime(12701): FATAL EXCEPTION: main
11-30 17:26:08.599 E/AndroidRuntime(12701): Process: com.mobile.myapp, PID: 12701
11-30 17:26:08.599 E/AndroidRuntime(12701): java.lang.IllegalArgumentException: View=com.android.internal.policy.PhoneWindow$DecorView{2fb63f2 V.E...... R......D 0,0-720,1440} not attached to window manager
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.view.WindowManagerGlobal.findViewLocked(WindowManagerGlobal.java:448)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.view.WindowManagerGlobal.removeView(WindowManagerGlobal.java:372)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.view.WindowManagerImpl.removeViewImmediate(WindowManagerImpl.java:116)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.app.Dialog.dismissDialog(Dialog.java:372)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.app.Dialog.dismiss(Dialog.java:351)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at com.luck.game.adv.ReaperAdvRequestUtilsKt$createInteractionExpressAdListener$1.run(SourceFile:141)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.os.Handler.handleCallback(Handler.java:815)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.os.Handler.dispatchMessage(Handler.java:104)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.os.Looper.loop(Looper.java:207)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at android.app.ActivityThread.main(ActivityThread.java:5821)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at java.lang.reflect.Method.invoke(Native Method)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:805)
11-30 17:26:08.599 E/AndroidRuntime(12701):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:695)

三 解决方法

解决方法也很简答,dismiss之前多加一个判断,activity是否已经销毁了。

if (dialog != null && dialog.isShowing && !activity.isDestroyed) {
    dialog.dismiss()
}

google推出DialogFragment,解决了生命周期的问题,而这个报错就是由生命周期引起的。所以后续的需求,还是尽量用google推荐的把,毕竟咱整个android都是它们造的,它自己肯定很了解这些缺陷,推荐的这些东西肯定是解决了一些问题的。

四 原因分析:

  1. 先看Dialog的源码,发现在dismiss之前有判断标志位,所以直观的认为即使activity销毁了,也是不会报错的:
/**
 * Dismiss this dialog, removing it from the screen. This method can be
 * invoked safely from any thread.  Note that you should not override this
 * method to do cleanup when the dialog is dismissed, instead implement
 * that in {@link #onStop}.
 */
@Override
public void dismiss() {
    if (Looper.myLooper() == mHandler.getLooper()) {
        dismissDialog();
    } else {
        mHandler.post(mDismissAction);
    }
}

void dismissDialog() {
    if (mDecor == null || !mShowing) {  //有判断标志位mShowing,所以认为不太容易出现异常,即使activity销毁了。
        return;
    }

    if (mWindow.isDestroyed()) {
        Log.e(TAG, "Tried to dismissDialog() but the Dialog's window was already destroyed!");
        return;
    }

    try {
        mWindowManager.removeViewImmediate(mDecor);  // 但是在这里,它实实在在的抛了异常。
    } finally {
        if (mActionMode != null) {
            mActionMode.finish();
        }
        mDecor = null;
        mWindow.closeAllPanels();
        onStop();
        mShowing = false;

        sendDismissMessage();
    }
}
  1. 其实在activity销毁之后,这个标志位并没有改变,所以通过判断这个标志位,是不足以避免抛出异常的。

  2. 从报错日志堆栈中可以看出,抛出异常的真正原因在
    E/AndroidRuntime(12701): at android.view.WindowManagerGlobal.findViewLocked(WindowManagerGlobal.java:448)
    看这里的源代码:

//WindowManagerGlobal.java
private int findViewLocked(View view, boolean required) {
    final int index = mViews.indexOf(view);
    if (required && index < 0) {
        throw new IllegalArgumentException("View=" + view + " not attached to window manager");
    }
    return index;
}

可见,是mViews里面已经没有Dialog的view了,所以抛出了异常。

  1. mViews里面什么时候移除的Dialog的view呢?在源码中搜索,发现有如下代码:
//WindowManagerGlobal.java
void doRemoveView(ViewRootImpl root) {
    synchronized (mLock) {
        final int index = mRoots.indexOf(root);
        if (index >= 0) {
            mRoots.remove(index);
            mParams.remove(index);
            final View view = mViews.remove(index);  //移除对应的view
            mDyingViews.remove(view);
        }
    }
    if (ThreadedRenderer.sTrimForeground && ThreadedRenderer.isAvailable()) {
        doTrimForeground();
    }
}

而在ViewRootImpl.java中,又有如下代码:

void doDie() {
    checkThread();
    if (LOCAL_LOGV) Log.v(mTag, "DIE in " + this + " of " + mSurface);
    synchronized (this) {
        if (mRemoved) {
            return;
        }
        mRemoved = true;
        if (mAdded) {
            dispatchDetachedFromWindow();
        }

        if (mAdded && !mFirst) {
            destroyHardwareRenderer();

            if (mView != null) {
                int viewVisibility = mView.getVisibility();
                boolean viewVisibilityChanged = mViewVisibility != viewVisibility;
                if (mWindowAttributesChanged || viewVisibilityChanged) {
                    // If layout params have been changed, first give them
                    // to the window manager to make sure it has the correct
                    // animation info.
                    try {
                        if ((relayoutWindow(mWindowAttributes, viewVisibility, false)
                                & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0) {
                            mWindowSession.finishDrawing(mWindow);
                        }
                    } catch (RemoteException e) {
                    }
                }

                mSurface.release();
            }
        }

        mAdded = false;
    }
    WindowManagerGlobal.getInstance().doRemoveView(this); //调用移除
}
  1. 确认ViewRootImpl.java的doDie()方法有在activity销毁时调用: 在activity销毁的时候,在一个自定义view里面的onDetachedFromWindow中,主动抛出一个空指针异常,并捕获和打印其堆栈:
12-24 10:22:48.160 W/System.err(30566): java.lang.NullPointerException: WeatherViewPager onDetachedFromWindow YQ
12-24 10:22:48.160 W/System.err(30566):     at net.test.myapp.WeatherViewPager.onDetachedFromWindow(WeatherViewPager.java:207)
12-24 10:22:48.160 W/System.err(30566):     at android.view.View.dispatchDetachedFromWindow(View.java:19690)
12-24 10:22:48.160 W/System.err(30566):     at android.view.ViewGroup.dispatchDetachedFromWindow(ViewGroup.java:3819)
12-24 10:22:48.160 W/System.err(30566):     at android.view.ViewGroup.dispatchDetachedFromWindow(ViewGroup.java:3811)
12-24 10:22:48.160 W/System.err(30566):     at android.view.ViewGroup.dispatchDetachedFromWindow(ViewGroup.java:3811)
12-24 10:22:48.160 W/System.err(30566):     at android.view.ViewRootImpl.dispatchDetachedFromWindow(ViewRootImpl.java:4204)
12-24 10:22:48.161 W/System.err(30566):     at android.view.ViewRootImpl.doDie(ViewRootImpl.java:7223)  //确认有调用
12-24 10:22:48.161 W/System.err(30566):     at android.view.ViewRootImpl.die(ViewRootImpl.java:7200)
12-24 10:22:48.161 W/System.err(30566):     at android.view.WindowManagerGlobal.removeViewLocked(WindowManagerGlobal.java:527)
12-24 10:22:48.161 W/System.err(30566):     at android.view.WindowManagerGlobal.removeView(WindowManagerGlobal.java:465)
12-24 10:22:48.161 W/System.err(30566):     at android.view.WindowManagerImpl.removeViewImmediate(WindowManagerImpl.java:126)
12-24 10:22:48.161 W/System.err(30566):     at android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:5122)
12-24 10:22:48.161 W/System.err(30566):     at android.app.servertransaction.DestroyActivityItem.execute(DestroyActivityItem.java:44)
12-24 10:22:48.161 W/System.err(30566):     at android.app.servertransaction.TransactionExecutor.executeLifecycleState(TransactionExecutor.java:176)
12-24 10:22:48.161 W/System.err(30566):     at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:97)
12-24 10:22:48.161 W/System.err(30566):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2065)
12-24 10:22:48.161 W/System.err(30566):     at android.os.Handler.dispatchMessage(Handler.java:107)
12-24 10:22:48.161 W/System.err(30566):     at android.os.Looper.loop(Looper.java:214)
12-24 10:22:48.161 W/System.err(30566):     at android.app.ActivityThread.main(ActivityThread.java:7656)
12-24 10:22:48.161 W/System.err(30566):     at java.lang.reflect.Method.invoke(Native Method)
12-24 10:22:48.161 W/System.err(30566):     at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:503)
12-24 10:22:48.161 W/System.err(30566):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:948)
  1. 完工,今日事今日毕。一直想着准备好了再记录,而总有不满意和不完美的地方,拖着事情不闭环,不如先记录下来,后续再慢慢完善和润色语言
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容