Android 窗口那些事儿

目录

1. 📂 前言

你,是否有过这些疑问?

2. 🔱 Window

2.1 认识 Window 的几个阶段

1)阶段一:Window 约等于 Activity

2)阶段二:Window 约等于 View

3)阶段三:Window 是个抽象封装概念

2.2 Android 中的 Window 定义

2.3 Window 到底是什么?

3. 💠 Window 相关

3.1 WindowManager

1)入参1:View

2)入参2:WindowManager.LayoutParams

3) 三个方法

3.2 DecorView 与 ViewRootImpl

3.3 WindowManagerService(WMS)

3.4 Layer

3.5 SurfaceFlinger

4. ⚛️ Window 深入

4.1 Activity/Window/View 的关系

4.2 Activity/Window/DecorView/ViewRootImpl 的创建时机

4.3 DecorView 什么时候被 WindowManager 添加到 Window 中

4.4 Android 多窗口原理

4.5 Window 点击与双击事件的区别

5. ✅ Window  底层

5.1 Window/View 添加过程

5.2 布局加载流程

5.3 View 绘制流程

1)addView 流程

2)performTraversals 流程

3)Measure、Layout、Draw 三大流程

5.4 View 绘制屏幕刷新

5.5 View 事件分发机制

这些疑问,你都解决了吗?


1. 📂 前言

你,是否有过这些疑问?

        Android 中窗口的定义是什么?

        Activity、Window、View 之间的关系?

        窗口到 View 的事件分发机制,是怎样的?

        WMS 是如何管理屏幕上显示的诸多窗口的?

        Android 是默认单窗口吗?多窗口怎么实现?

        SurfaceFlinger 合成的 Layer 与窗口是一一对应的吗?

        SurfaceFlinger、Layer、Window,以及 WMS 的联系?

        点击的事件分发与双击的事件分发,为什么走的是不一套机制?

下面,就一起去探讨下这些问题吧。有些理解只是抛砖引玉,并未完全解答,还请见谅。

2. 🔱 Window

        首先这样一个场景,下图包含:Activity、Dialog、Toast。 

  • 问:图里有几个窗口?

  • 答:三个,Activity 应用窗口,Dialog 子窗口,Toast 系统窗口。

        我们知道,Android 中的窗口,其实就是由以上三种组成,分别是应用 Window、子 Window、系统 Window。

2.1 认识 Window 的几个阶段

1)阶段一:Window 约等于 Activity

        刚接触 Android 时,认为 Activity 就是一个显示文本、图片的界面,所以 Window 就约等于 Activity。

2)阶段二:Window 约等于 View

        在接触 Window、WMS 概念后,知道 Activity 内部持有 Window 对象,而 Window 实现类 PhoneWindow 内部持有 DecorView 作为根布局,开发人员编写的 ContentView 会添加到 DecorView 中,所以 Window 就约等于 View。

3)阶段三:Window 是个抽象封装概念

        在深入 Window、WMS 源码后,我们知道:

  • 每一个 Window 都对应着一个 View 和 一个 ViewRootImpl;

  • Window 作为 View 对象的容器,以 View 的方式存在,所以 Window 可以称之为 View 的直接管理者;

  • Window 并不是实际存在的,它表示一个窗口的概念,也是一个抽象的概念,封装了对窗口的操作逻辑。

2.2 Android 中的 Window 定义

        接下来,我们看看源码中 Window 类的注释:

/**
 * Abstract base class for a top-level window look and behavior policy.  An
 * instance of this class should be used as the top-level view added to the
 * window manager. It provides standard UI policies such as a background, title
 * area, default key processing, etc.
 *
 * <p>The only existing implementation of this abstract class is
 * android.view.PhoneWindow, which you should instantiate when needing a
 * Window.
 */
public abstract class Window {

}

        结合 Window 类的注释,我们知道 Window 就是一层封装,提供通用页面模板,并不是真正的窗口。

2.3 Window 到底是什么?

        Window 是一个抽象的概念,对应屏幕显示图像的一块区域,实际是 View。比如:前言中的图例情景,共包含3个 Window,分为3块区域,对应3个具体 View,当在 Dialog 的 Window 中添加、更新与删除 View,是不会影响到 Activity 和 Toast 的。

        微观角度,Window 就是 View;宏观角度,Window 是 WMS 窗口体系下的最小单位。

        其实,根本就没有具体的 Window,只有具体的 View。

3. 💠 Window 相关

3.1 WindowManager

        WindowManager 是一个接口,继承自 ViewManager 接口,它的实现类为 WindowManagerImpl,WindowManagerImpl 通过桥接模式,将所有操作全部委托给 WindowManagerGlobal 来实现。

        WindowManager 是我们访问 Window 的入口,使用 WindowManager 对 Window 进行添加、更新和删除,具体工作则由 WMS 来处理,WindowManager 和 WMS 通过 Binder 来进行跨进程通信。

        接下来,让我们先从 WindowManager.addView(view, layoutParams) 方法的2个入参开始聊起。

1)入参1:View

        View 表示需要在屏幕显示的内容,它是具体的。

2)入参2:WindowManager.LayoutParams

        LayoutParams 对内容进行约束,包括宽高、位置、类型和 flag 等。

LayoutParams.type

        表示 Window 的类型,Window 共有三种类型,分别是应用 Window、子 Window、系统 Window。

  1. 应用 Window:1 ~ 99,如:Activity 的 type 为2;

  2. 子 Window:1000 ~ 1999,如:PopupWindow 的 type 默认为1000,Dialog 的 type 为1003;

  3. 系统 Window:2000~ 2999,如:状态栏 type 为2000,Toast  type 为2005,悬浮窗口 type 为2038。

        数值越大层级越高,层级高覆盖层级低的,一般通过常量设置,系统 Window 需要申请权限。

LayoutParams.flags

        设置 Window 相关的属性,常用:FLAG_NOT_FOCUSABLE、FLAG_NOT_TOUCH_MODA、FLAG_SHOW_WHEN_LOCKED 等。

  1. FLAG_NOT_FOCUSABLE:表示窗口不需要获取焦点,也不需要接收各种事件,此属性会同时启动FLAG_NOT_TOUCH_MODAL,最终的事件会传递给下层拥有焦点的 Window;

  2. FLAG_NOT_TOUCH_MODAL:将 Window 区域以外的单击事件传递给下层的 Window, 当前 Window 内的单击事件自己处理, 一般都要开启此事件,否则其他 Window 无法收到单击事件;

  3. FLAG_SHOW_WHEN_LOCKED:可以将 Window 显示在锁屏的界面上。

3) 三个方法

        WindowManager 常用的有三个方法,addView、updateView 和removeView,这三个方法定义在 ViewManager 中,而 WindowManager 继承自 ViewManager。

package android.view;

public interface ViewManager {
    public void addView(View view, ViewGroup.LayoutParams params);
    public void updateViewLayout(View view, ViewGroup.LayoutParams params);
    public void removeView(View view);
}

3.2 DecorView 与 ViewRootImpl

        DecorView 是 FrameLayout 的子类,被认为是 Android 视图树的根节点视图。

        ViewRootImpl 是一个抽象的概念,相比于 DecorView 这种“实体”根节点,它是一个虚拟的根节点。

        Measure-Layout-Draw 三大 View 绘制流程、触屏和按键等 Input 事件传递,以及 Insets 的更新、大部分需要遍历 View 层级结构的流程,起点都是在 ViewRootImpl。

        简单来说,ViewRootImpl 不仅是 View 和 WM 的桥梁,也是事件分发的桥梁,是视图系统的核心。

3.3 WindowManagerService(WMS)

        WMS 是 Android 系统中的重要服务,管理所有窗口,也是输入事件的中转站。

        对 WMS 来说,一个窗口就是一个可以用来显示的 View 类,一个通过 WindowManagerGlobal.addView() 添加的 View。

        View 本身并不能直接从 WMS 中接收消息,而是通过实现了 IWindow 接口的 ViewRootImpl.W 类来实现。

3.4 Layer

        Layer 是一个比 Window 更底层的实现,代表屏幕上一块显示内容的区域。

        一个 Window 就是一个 Surface,对应一个 Layer。

3.5 SurfaceFlinger

        SurfaceFlinger 作用是合成所有 Layer 并送显。在 App 请求创建 Surface 时,SurfaceFlinger 会创建一个 Layer,在拿到从 WMS 传过来的 Window 宽高、位置,以及 App 提供需要绘制的 View 后合成该 Layer,然后送给屏幕进行显示。

        在 Android 平台上创建的每个 Window 都由 Surface 提供支持,所有被渲染的可见 Surface 都被 SurfaceFlinger 合成到屏幕。

 

4. ⚛️ Window 深入

4.1 Activity/Window/View 的关系

  1. Activity:是最上层的封装,屏蔽复杂的系统实现细节,抽象出 UI 生命周期,方便开发人员专注于界面编写,持有 Window;

  2. Window:也是一层封装,提供通用页面模板,作为视图承载器,负责视图控制,持有 DecorView,DecorView 就是 View 的根布局;

  3. View:就是视图,在 Activity.setContentView 中将 View 视图添加到 DecorView 中,一个页面通用且不变的部分交给 PhoneWindow 实现,变化的部分交给 View,让开发人员能够自由定制。

        Activity 很庞大,通过采用单一职责原则,把 View 相关的代码从 Activity 中剥离到 Window 中去。

4.2 Activity/Window/DecorView/ViewRootImpl 的创建时机

  1. Activity:在 ActivityThread 的 performLaunchActivity 方法中,通过 Instrumentation 在内部使用类加载器创建 Activity 实例;

  2. Window:在 ActivityThread 的 performLaunchActivity 方法中,调用 Activity 的 attach 方法 new 一个 PhoneWindow 然后赋值给 mWindow 成员变量;

  3. DecorView:在 Activity 的 onCreate 方法中调用 setContentView 方法,最后会调用 PhoneWindow 的 generateDecor 方法 new 一个 DecorView 然后赋值给 mDecor 成员变量;

  4. ViewRootImpl

    1. 在 ActivityThread 的 handleResumeActivity 方法中,内部通过 WindowManagerGlobal 的 addView 方法 new 一个 ViewRootImpl 然后赋值给局部变量 root(ViewRootImpl);

    2. 通过 addView 方法,我们也注意到,一个 Window 对应一个 ViewRootImpl;

    3. 此时也会调用 setView 方法与 DecorView 进行绑定:ViewRootImpl 会声明 mView 成员变量,并在 setView 方法中给 mView 对象赋值;

    4. 和 PhoneWindow 一样会持有 DecorView,但 DecorView 的创建还是在 PhoneWindow 的;

4.3 DecorView 什么时候被 WindowManager 添加到 Window 中

  1. 即使 Activity 布局已成功添加到 DecorView 中,但 DecorView 此时还没有添加到 Window 中;

  2. 在 ActivityThread 的 handleResumeActivity 方法中,首先会调用 Activity.onResume 方法,接着调用 Activity.makeVisible 方法,在 makeVisible() 中完成 DecorView 的添加和显示。

4.4 Android 多窗口原理

        简单来说,多窗口框架的核心思想是分栈设置栈边界

  1. 分栈:在 Android 系统中,启动一个 Activity 之后,必定会将此 Activity 存放于某一个 Stack,Android 为了支持多窗口,在运行时创建了多个 Stack;

  2. 栈边界:在多窗口框架中,通过设置 Stack 的边界来控制里面每个 Task 的大小,最终 Task 的大小决定了里面的 Activity 的窗口大小。

4.5 Window 点击与双击事件的区别

  1. 单击走的是 MotionEvent、双击走的是 KeyEvent;

  2. MotionEvent 会返回位置坐标,KeyEvent 不会上报位置坐标。

5. ✅ Window  底层

5.1 Window/View 添加过程

  1. Window/View

  2. WindowManager.addView()

  3. WindowManagerImpl.addView()

  4. WindowManagerGlobal.addView()

  5. ViewRootImpl.setView()

  6. WindowManagerService

        WindowMangerImpl 通过桥接模式,将所有操作全部委托给 WindowManagerGlobal 来实现。

        不管是 Activity 窗口还是非 Activity 窗口,最终都要通过 ViewRootImpl.setView 向 WMS 注册窗口。

5.2 布局加载流程

  1. setContentView 中通过 LayoutInflate.inflate 加载对应布局;

  2. inflate 方法中首先调用 Resources.getLayout 通过 pull 方式 IO 加载 Xml 布局解析器到内存中;

  3. createViewFromTag 根据 xml 的 Tag 标签反射创建 View 到内存;

  4. 递归构建其中子 View,并将子 View 添加到父 ViewGroup 中。

        Android 这套布局加载流程的性能瓶颈:布局文件解析中的 IO 过程;创建 View 对象时的反射过程。

5.3 View 绘制流程

1)addView 流程
  1. ActivityThread.handleResumeActivity;

  2. WindowManagerImpl.addView;

  3. WindowMangerGlobel.addView;

  4. ViewRootImpl.setView;

  5. ViewRootImpl.scheduleTraversals;

  6. ViewRootImpl.doTraveral;

  7. ViewRootImpl.performTraversals。

2)performTraversals 流程
  1. performMeasure—DecorView.measure—DecorView.onMeasure—View.measure;

  2. performLayout—DecorView.layout—DecorView.onLayout—View.layout;

  3. performDraw—DecorView.draw—DecorView.onDraw—View.draw。

3)Measure、Layout、Draw 三大流程
  1. Context.startActivity;

  2. ActivityThread.handleLaunchActivity,执行 onCreate,完成 DecorView 和 Activity 的创建;

  3. ActivityThread.handleResumeActivity,执行 onResume,完成 DecorView 添加到 WindowManager;

  4. ViewRootImpl.performTraversals(),测量、布局、绘制, 从 DecorView 自上而下遍历整个 View 树。

5.4 View 绘制屏幕刷新

  1. CPU:主要负责 Measure、Layout、Record、Execute 数据计算工作;

  2. GPU:负责栅格化(向量图形格式表示的图像转换成位图用于显示器)、渲染,渲染好后放到 buffer(图像缓冲区)里存起来;

  3. Display:屏幕或显示器会以一定的帧率刷新,每次刷新时,就会从缓存区将图像数据读取显示出来,如果缓存区没有新数据,就一直用旧数据,这样屏幕看起来就没有变。

        CPU 准备数据,通过 Driver 层把数据交给 GPU 渲染,Display 负责消费显示内容。

5.5 View 事件分发机制

  1. 定义:将事件(MotionEvent)传递到某个具体的 View & 处理的整个过程;

  2. ViewRootImpl—DecorView—Activity—Window—DecorView(ViewGroup)—View:事件先到 DecorView 然后到 Window。当屏幕被触摸时,input 系统事件从 Native 层分发到 Framework 层,然后到 ViewRootImpl 的 DecorView,进一步到 Activity -> PhoneWindow -> DecorView,最后到 View;

  3. DecorView 怎么将事件分发给 Activity 的?

                a、在 DecorView 的 dispatchKeyEvent 方法内部通过 Window 获取 callback,然后执行 callback 的 dispatchKeyEvent ;

                b、Activity 本身实现了 Window.Callback 接口,并设置给了 Window 的 callback,所以这里的 callback 其实就是 Activity,这样事件就传递到了 Activity;

        为什么 DecorView 走了两遍:主要原因就是解耦。

  1. ViewRootImpl 并不知道 Activity 的存在,它只是持有了 DecorView,所以先传给了 DecorView,而 DecorView 知道有 Activity,所以传给了 Activity;

  2. Activity 也不知道有 DecorView ,它只是持有 PhoneWindow,于是这样一段调用链就形成了。


 

这些疑问,你都解决了吗?

        Android 中窗口的定义是什么?

        Activity、Window、View 之间的关系?

        窗口到 View 的事件分发机制,是怎样的?

        WMS 是如何管理屏幕上显示的诸多窗口的?

        Android 是默认单窗口吗?多窗口怎么实现?

        SurfaceFlinger 合成的 Layer 与窗口是一一对应的吗?

        SurfaceFlinger、Layer、Window,以及 WMS 的联系?

        点击的事件分发与双击的事件分发,为什么走的是不一套机制?


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/520191.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

list的使用

前言 我们前面已经对string和vector进行了学习使用&#xff0c;以及对他们的底层进行了模拟实现&#xff01;本期我们继续学习STL的另外一个容器---list。 本期内容介绍 什么是list&#xff1f; list的常用接口 什么是list? 还是来看看官方的文档说明&#xff01; 这里通过…

[蓝桥杯 2017 国 C] 合根植物

[蓝桥杯 2017 国 C] 合根植物 题目描述 w 星球的一个种植园&#xff0c;被分成 m n m \times n mn 个小格子&#xff08;东西方向 m m m 行&#xff0c;南北方向 n n n 列&#xff09;。每个格子里种了一株合根植物。 这种植物有个特点&#xff0c;它的根可能会沿着南北…

【MySQL】增删改查操作(基础)

文章目录 1、新增操作&#xff08;Create&#xff09;1.1单行数据全列插入1.2多行数据指定列插入 2、查询操作&#xff08;Retrieve&#xff09;2.1全列查询2.2指定列查询2.3指定列查询2.4别名&#xff08;as&#xff09;2.5去重&#xff08;distinct&#xff09;2.6排序&#…

数据结构—图

图的基本概念 图就是由顶点的有穷非空集合和顶点之间的边组成的集合。通常表示为&#xff1a;G(V,E)&#xff0c;其中&#xff0c;G 表示一个图&#xff0c;V 表示顶点的集合&#xff0c;E 表示边的集合。 顶点 图中的数据元素&#xff0c;我们称之为顶点&#xff0c;图至少有…

常见现代卷积神经网络(Pytorch 09)

本章将介绍现代的 卷积神经网络架构&#xff0c;许多现代卷积神经网络的研究都是建立在这一章的基础上的。在本章中的每一个模型都曾一度占据主导地位&#xff0c;其中许多模型都是 ImageNet竞赛 的优胜者。ImageNet竞赛自2010年以来&#xff0c;一直是计算机视觉中监督学习进展…

面试题——JVM老年代空间担保机制(我的想法)

这里借用一下人家的图&#xff0c;来说一下我的想法&#xff0c;嘻嘻。。。。 原文链接&#xff1a;一道面试题&#xff1a;JVM老年代空间担保机制-CSDN博客? 嗯&#xff0c;我觉得老年代担保机制的主要作用就是避免频繁触发FULL GC&#xff0c;这其实也是因为年轻代Minor GC…

Java项目:基于Springboot+vue社区医院管理系统设计与实现(源码+数据库+毕业论文)

一、项目简介 本项目是一套基于Springbootvue社区医院管理系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、功能…

数据结构之顺序表的相关知识点及应用

个人主页&#xff08;找往期文章包括但不限于本期文章中不懂的知识点&#xff09;&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 目录 顺序表的概念及结构 顺序表的分类 顺序表的实现 在顺序表中增加数据 在顺序表中删除数据 在顺序表中查找数据 顺序表源码 顺序表的概念…

浮动辊位移测量功能块(CODESYS ST代码)

1、张力测量+标定(ST代码) 张力测量+标定(ST代码)_动态舞轮控制张力-CSDN博客文章浏览阅读804次。跳舞轮对应张力调节范围,我们可以通过改变气缸的气压方式间接改变,张力跳舞轮在收放卷闭环控制上的详细应用,可以参看下面的文章链接,这里我们主要讨论精密可调气阀的模拟量…

Java | Leetcode Java题解之第6题Z字形变换

题目&#xff1a; 题解&#xff1a; class Solution {public String convert(String s, int numRows) {int n s.length(), r numRows;if (r 1 || r > n) {return s;}int t r * 2 - 2;int c (n t - 1) / t * (r - 1);char[][] mat new char[r][c];for (int i 0, x …

[Spring Cloud] gateway全局异常捕捉统一返回值

文章目录 处理转发失败的情况全局参数同一返回格式操作消息对象AjaxResult返回值状态描述对象AjaxStatus返回值枚举接口层StatusCode 全局异常处理器自定义通用异常定一个自定义异常覆盖默认的异常处理自定义异常处理工具 在上一篇章时我们有了一个简单的gateway网关 [Spring C…

比selenium体验更好的ui自动化测试工具: cypress介绍

话说 Cypress is a next generation front end testing tool built for the modern web. And Cypress can test anything that runs in a browser.Cypress consists of a free, open source, locally installed Test Runner and a Dashboard Service for recording your tests.…

leetcode077——排序链表

题目&#xff1a; 给定链表的头结点 head &#xff0c;请将其按 升序 排列并返回 排序后的链表 。 示例 1&#xff1a; 输入&#xff1a;head [4,2,1,3] 输出&#xff1a;[1,2,3,4] 思路&#xff1a; 1.找链表中点【使用快慢指针 慢指针每次移动一步&#xff0c;快指针每…

基于单片机12864的出租车计价器设计

**单片机设计介绍&#xff0c;基于单片机12864的出租车计价器设计 文章目录 一 概要二、功能设计三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机和12864液晶显示屏的出租车计价器设计&#xff0c;主要是利用单片机的强大控制能力和液晶显示屏的直观显示特性&…

牛客网BC-125 序列中整数去重复(难题讲解)

题目如下 --------------------------------------------------------------------------------------------------------------------------------- 题目讲解&#xff08;思路&#xff09; -------------------------------------------------------------------------------…

单一职责原则

1.1 阅读干吗不直接用手机&#xff1f; 电子阅读器比较专注&#xff0c;而手机功能比较多&#xff0c;影响专注。 1.2 手机不纯粹 手机确实很方便。但是现在的手机就是一台小型智能电脑。它不仅能打电话&#xff0c;还能听音乐、看电影电视、与个人交流、与一群人群聊&#…

基于java+SpringBoot+Vue的大学生入学审核系统设计与实现

基于javaSpringBootVue的大学生入学审核系统设计与实现 开发语言: Java 数据库: MySQL技术: SpringBoot VUE工具: IDEA/Eclipse、Navicat、Maven 系统展示 前台展示 入学办理模块&#xff1a;学生可以提交入学申请并跟踪入学办理进度。 后台展示 学生管理模块&#xff1…

【Docker系列】在 Linux 上安装 Docker Compose 的简明步骤

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

前端学习之DOM编程案例:抽奖案例

代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>抽奖案例</title><style>*{margin: 0;padding: 0;}</style> </head> <body><div id"container"&g…

【放假第3天】幻兽帕鲁 雾锁王国 我的世界 游戏云服务器选购指南 附最新价格对比表 新手、小白秒懂

更新日期&#xff1a;4月6日&#xff08;半年档 价格回调&#xff0c;京东云采购季持续进行&#xff09; 本文纯原创&#xff0c;侵权必究 【云服务器推荐】价格对比&#xff01;阿里云 京东云 腾讯云 选购指南视频截图 《最新对比表》已更新在文章头部—腾讯云文档&#xf…