美文网首页面试题
Android高频面试专题 - 提升篇(一)Window、Vie

Android高频面试专题 - 提升篇(一)Window、Vie

作者: Android扫地僧 | 来源:发表于2020-02-25 12:39 被阅读0次

    本篇主要介绍Window相关面试要点,常见Window属性,Window、Activity、View之间的关系。更多其他完整面试专题,请关注公众号获取.

    1、Window是什么?

    表示一个窗口的概念,是所有View的直接管理者,任何视图都通过Window呈现(点击事件由Window->DecorView->View; Activity的setContentView底层通过Window完成)

    • Window是一个抽象类,唯一实现类是PhoneWindow

    • 创建Window需要通过WindowManager创建,WindowManager是外界访问Window的入口

    • Window具体实现位于WindowManagerService中,WindowManager和WindowManagerService的交互是通过IPC完成

    2、Window的内部机制

    • Window和View通过ViewRootImpl建立联系

    • Window并不是实际存在的,而是以View的形式存在

    • WindowManager的三个接口方法也是针对View的

    • 实际使用中无法直接访问Window,必须通过WindowManager

    • View是视图的呈现方式,但是不能单独存在,必须依附在Window这个抽象的概念上

    • WMS把所有的用户消息发给View/ViewGroup,但是在View/ViewGroup处理消息的过程中,有一些操作是公共的, Window把这些公共行为抽象出来, 这就是Window。

    3、Window有哪几种类型

    FrameWork定义了三种窗口类型,三种类型定义在WindowManager,通过LayoutParams.type设置。

    • 应用窗口,对应于一个Activity。加载Activity由AmS完成,创建一个应用窗口只能在Activity内部完成(层级1~99)。

    • 子窗口,必须依附于任何类型的父窗口(层级1000~1999)。

    • 系统窗口,不需要对应任何Activity,如:状态栏,导航栏,普通应用程序不能创建系统窗口,必须要有系统应用权限.(层级2000~2999)。

    WindowManager为这个三类进行了细化,把每一种类型都有int常量标识,WmS进行窗口叠加的时候会按照该int常量的大小分配不同层,int值越大层位置越靠上面。

    4.WindowManager的三个主要功能:****添加、更新、删除View

    public interface ViewManager{
        public void addView(View view, ViewGroup.LayoutParams params); //添加View
        public void updateViewLayout(View view, ViewGroup.LayoutParams params); //更新View
        public void removeView(View view); //删除View
    }
    

    5.Activity中setContentView()发生了什么

    Activity中setContentView()后实际通过getWindow().setContentView()交由PhoneWindow处理,PhoneWindow中主要做两件事,通过installDecor()初始化mDecor(DecorView)和generateLayout()来初始化mContentParent(ViewGroup), 然后通过inflate将我们的setContentView传入的View或者layout布局文件填充到这个mContentParent中,后面会讲到具体细节。其中在generateLayout()实际上就是在根据我们requestFeature设置的style(如FULL_SCREEN,NO_ACTION_BAR)加载对应的布局容器(这里也可以解释为什么我们getWindow.requestFeature时必须在setContentView()之前),这个容器中会有一个id为content的FrameLayout,这个FrameLayout就是上面所说的mContentParent, 也就是说我们setContentView()最终是设置到这里。

    完整流程可以参考从Activity创建到View呈现中间发生了什么

    6.DecorView是什么?

    DecoreView本质就是一个FrameLayout,是Activity中的顶级View,如果我们不设置任何主题style,默认加载的DecorView会addView以下布局文件

    image

    7.Activity、Window、DecorView、View之间的关系

    每个Activity 包含了一个Window 对象,这个对象是由PhoneWindow做的实现。而 PhoneWindow 将DecorView作为了一个应用窗口的根View,这个DecorView 又把屏幕划分为了两个区域:一个是 TitleView,也就是ActionBar或者TitleBar,一个是 ContentView,而我们平时在 Xml 文件中写的布局正好是展示在 ContentView 中的。

    image

    8.DecorView何时才被WindowManager真正添加到Window中?

    • 即使Activity的布局已经成功添加到DecorView中,DecorView此时还没有添加到Window中

    • ActivityThread的handleResumeActivity方法中,首先会调用Activity的onResume方法,接着调用Activity的makeVisible()方法

    • makeVisible()中通过WindowManager.addView()完成了DecorView的添加和显示两个过程

    void makeVisible() {
      //1. 将`DecorView`添加到`Window`中(通过WindowManager)
        if (!mWindowAdded) {
            ViewManager wm = getWindowManager();
            wm.addView(mDecor, getWindow().getAttributes());
            mWindowAdded = true;
        }
      //2. 将DecorView显示出来
        mDecor.setVisibility(View.VISIBLE);
    }
    

    9、ViewRoot是什么

    ViewRoot对应ViewRootImpl类,它是连接WMS和DecorView的纽带,但它却并不属于View树的一份子,并不是View的子类也不是View的父类,但它实现了ViewParent接口,所以可以作为名义上的View的父视图。

    WindowManager.addView()内部实际是由WindowManagerGlobal完成的,WindowManagerGlobal中有三个列表,一个是保存View的mViews列表,一个是保存ViewRootImpl的mRoots列表,一个是保存WindowManager.LayoutParams的mParams列表,WindowManager每一次addView()都会创建一个对应的ViewRootImpl,在调用ViewRoot.setView后将decorView交给ViewRootImpl。ViewRootImpl中调用performTraversals方法,然后便开始测量布局绘画了,界面才得以显示出来,这就是View的绘制流程起点。

    10、Token是什么?

    类型为IBinder,是一个Binder对象。

    主要分两种Token:

    指向Window的token: 主要是实现WmS和应用所在进程通信。

    指向ActivityRecord的token: 主要是实现WMS和AMS通信的。

    11、Token的使用场景?

    Activity创建时,AMS中需要根据Token去找到对应的ActivityRecord。

    Popupwindow的showAtLocation第一个参数需要传入View,这个View就是用来获取Token的。

    Android 5.0新增空间SnackBar同理也需要一个View来获取Token

    12、WindowSession是什么

    在WindowManager的addView中会创建ViewRootImpl,内部会通过WMS去获取WindowSession

    WindowSession的类型是IWindowSession,本身是Binder对象,真正实现类是Session

    作用:

    表示一个Active Client Session

    每个进程一般都有一个Session对象

    用于WindowManager交互

    13、Dialog的Window创建过程

    • 创建Window——同样是通过PolicyManager的makeNewWindow方法完成,与Activity创建过程一致

    • 初始化DecorView并将Dialog的视图添加到DecorView中——和Activity一致(setContentView)

    • 将DecorView添加到Window中并显示——在Dialog的show方法中,通过WindowManager将DecorView添加到Window中(mWindowManager.addView(mDecor, 1))

    • Dialog关闭时会通过WindowManager来移除DecorView:mWindowManager.removeViewImmediate(mDecor)

    • Dialog必须采用Activity的Context,因为有应用token(Application的Context没有应用token),也可以将Dialog的Window通过type设置为系统Window(SYSTEM_ALERT,需要申请权限)就不再需要token。

    14、Window常见FLAG属性

    image image

    更多进阶资源,请关注下方二维码,或直接微信搜索“Android扫地僧”


    Android扫地僧

    相关文章

      网友评论

        本文标题:Android高频面试专题 - 提升篇(一)Window、Vie

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