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

点击上方 Android扫地僧 ,选择 星标 公众号

重磅资源、干货分享,快上车!

本篇主要介绍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以下布局文件
Android高频面试专题 - 提升篇(一)Window、View、Activity_第1张图片

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

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

Android高频面试专题 - 提升篇(一)Window、View、Activity_第2张图片

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属性

Android高频面试专题 - 提升篇(一)Window、View、Activity_第3张图片
Android高频面试专题 - 提升篇(一)Window、View、Activity_第4张图片


Android高频面试专题 - 提升篇(一)Window、View、Activity_第5张图片

你可能感兴趣的:(面试专题#提升篇)