美文网首页
每日一题:为什么Dialog不能用Application的Con

每日一题:为什么Dialog不能用Application的Con

作者: 代码我写的怎么 | 来源:发表于2023-06-19 15:47 被阅读0次

    在android面试中,我们常会遇到Framework面试相关问题,而今天要分享的就是为什么Dialog不能用Application的Context?

    其主要考察的是程序员是否了解Dialog的运行机制?

    一般遇到此类问题我们可以从以下两个方面去回答:

    1. Window、WindowManager、WindowMangerService之间的关系
    2. Dialog使用Activity的Token的原因

    问题正解:

    1. 首先我们看一下如果用Application的Context出现什么状况?
     Caused by: android.view.WindowManager$BadTokenException: Unable to add window -- token null is not valid; is your activity running?
            at android.view.ViewRootImpl.setView(ViewRootImpl.java:907)
            at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:387)
            at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:95)
            at android.app.Dialog.show(Dialog.java:342)
    
    • 从上面代码我们可以得出,不能把Dialog添加到Window上,因为Token为空是非法的,末尾一句,“is your activity running?”。我们可以猜测,Dialog需要一个合法的Token,而且这个在Activity运行后是可以获得的。
    1. Window、WM、WMS、Token的概念?
    • Window:Window是窗口的意思,对应屏幕上的一块显示区域,它的实现类是PhoneWindow。 Window有一个属性值Type(应用窗口、子窗口、系统窗口);应用窗口有Activity,子窗口有PopupWindow、ContextMenu、OptionMenu;系统窗口有Toast和系统警告提示框。
    • WM:WM全称是WindowManager,WindowManager是在应用与Window之间管理接口,像窗口顺序、消息等。
    • WMS:WMS全称是WindowManagerService,WindowManagerService是窗口的管理者,它负责窗口的启动、添加和删除。另外窗口的大小和层级也是由 WMS 进行管理的。
    • Token:这里讲到的Token主要是指窗口令牌(Window Token),是一种特殊的Binder令牌,WMS用它唯一标识系统中的一个窗口
    1. Dialog的窗口属于什么类型?
    • Dialog是拥有一个PhoneWindow的实例,是应用窗口类型TYPE_APPLICATION。
    1. 来看下Dialog的构造方式
    Dialog(@NonNull Context context, @StyleRes int themeResId, boolean createContextThemeWrapper) {
            if (createContextThemeWrapper) {
                if (themeResId == Resources.ID_NULL) {
                    final TypedValue outValue = new TypedValue();
                    context.getTheme().resolveAttribute(R.attr.dialogTheme, outValue, true);
                    themeResId = outValue.resourceId;
                }
                mContext = new ContextThemeWrapper(context, themeResId);
            } else {
                mContext = context;
            }
    
            mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
    
            final Window w = new PhoneWindow(mContext);
            mWindow = w;
            w.setCallback(this);
            w.setOnWindowDismissedCallback(this);
            w.setOnWindowSwipeDismissedCallback(() -> {
                if (mCancelable) {
                    cancel();
                }
            });
            w.setWindowManager(mWindowManager, null, null);
            w.setGravity(Gravity.CENTER);
    
            mListenersHandler = new ListenersHandler(this);
        }
    
    • 从上述代码我们可以得出,Dialog在构造方法中设置WindowManager传入的appToken为空,那Dialog的appToken在什么时候获取的了。
    • 如果用Application或者Service的Context去获取这个WindowManager服务的话,会得到一个WindowManagerImpl的实例,这个实例里token也是空的。
    • 如果我们使用Activity作为Context,会拿到Activity的mWindowManager,这个mWindowManager在Activity的attach方法被创建,Token指向此Activity的Token

    今日分享到此结束,下期更精彩~

    关注个人简介,面试不迷路~

    相关文章

      网友评论

          本文标题:每日一题:为什么Dialog不能用Application的Con

          本文链接:https://www.haomeiwen.com/subject/teyeydtx.html