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 的联系?

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


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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于深度学习的危险物品检测系统(网页版+YOLOv8/v7/v6/v5代码+训练数据集)

摘要&#xff1a;本文详细介绍基于YOLOv8/v7/v6/v5的危险物品检测技术。主要采用YOLOv8技术并整合了YOLOv7、YOLOv6、YOLOv5的算法&#xff0c;进行了细致的性能指标对比分析。博客详细介绍了国内外在危险物品检测方面的研究现状、数据集处理方法、算法原理、模型构建与训练代码…

保健品wordpress外贸模板

保健品wordpress外贸模板 健康保养保健品wordpress外贸模板&#xff0c;做大健康行业的企业官方网站模板。 https://www.jianzhanpress.com/?p3514

蓝桥杯刷题第八天(dp专题)

这道题有点像小学奥数题&#xff0c;解题的关键主要是&#xff1a; 有2种走法固走到第i级阶梯&#xff0c;可以通过计算走到第i-1级和第i-2级的走法和&#xff0c;可以初始化走到第1级楼梯和走到第2级楼梯。分别为f[1]1;f[2]1(11)1(2)2.然后就可以循环遍历到后面的状态。 f[i…

XenCenter 2024 创建一个虚拟机

前言 实现&#xff0c;创建一个虚拟机&#xff0c;内存&#xff0c;cpu&#xff0c;磁盘&#xff0c;名称&#xff0c;网卡&#xff0c;配置 Xen Center 2024 download 创建虚拟机 选择系统类型 定义虚拟机名称 选择ISO镜像库 选择主服务器 分配虚拟机内存&#xff0c;cpu资源…

观察和配置MAC地址表

目录 原理概述 实验目的 实验内容 实验拓扑 ​编辑1&#xff0e;基本配置 2.观察正常状态时的MAC地址表 4.配置静态MAC地址表项 原理概述 MAC 地址表是交换机的一个核心组成部分&#xff0c;交换机主要是根据 MAC 地址表来进行帧的转发的。交换机对帧的转发操作行为一共有…

OpenHarmony实战:轻量带屏解决方案之恒玄芯片移植案例

本文章基于恒玄科技 BES2600W 芯片的欧智通 Multi-modal V200Z-R 开发板&#xff0c;进行轻量带屏开发板的标准移植&#xff0c;开发了智能开关面板样例&#xff0c;同时实现了 ace_engine_lite、arkui_ui_lite、aafwk_lite、appexecfwk_lite、HDF 等部件基于 OpenHarmony Lite…

集成电路企业tapeout,如何保证机台数据准确、完整、高效地采集?

Tapeout即流片&#xff0c;集成电路行业中将CDS最终版电路图提交给半导体制造厂商进行物理生产的过程。在芯片设计与制造的流程中&#xff0c;Tapeout是非常重要的阶段&#xff0c;包括了布局&#xff08;Layout&#xff09;、连线&#xff08;Routing&#xff09;、分析&#…

Linux 线程互斥、互斥量、可重入与线程安全

目录 一、线程互斥 1、回顾相关概念 2、抢票场景分析代码 多个线程同时操作全局变量 产生原因 如何解决 二、互斥量 1、概念 2、初始化互斥量&#xff1a; 方法1&#xff1a;静态分配 方法2&#xff1a;动态分配 3、销毁互斥量&#xff1a; 4、加锁和解锁 示例抢…

企业如何设计和实施有效的网络安全演练?

现实世界中&#xff0c;武装部队一直利用兵棋推演进行实战化训练&#xff0c;为潜在的军事冲突做准备。随着当今的数字化转型&#xff0c;同样的概念正在以网络安全演习的形式在组织中得到应用&#xff0c;很多企业每年都会基于合理的网络攻击场景和事件响应做一些测试和模拟。…

鸿蒙南向开发案例:【智能养花机】

样例简介 智能养花机通过感知花卉、盆栽等植宠生长环境的温度、湿度信息&#xff0c;适时为它们补充水分。在连接网络后&#xff0c;配合数字管家应用&#xff0c;用户可远程进行浇水操作。用户还可在应用中设定日程&#xff0c;有计划的按日、按周进行浇水。在日程中用户可添…

uniapp-打包app-图标配置

依次找到manifest->App图标配置&#xff0c;然后点击浏览&#xff0c;从本地文件夹中选择你们项目的logo&#xff0c;然后点击自动生成所有图标并替换&#xff0c;即可&#xff1a;

uniapp创建opendb-city-china Schema文件后,如何导入城市的数据?

1.点击opendb-city-china后面的详情&#xff0c;进入到gitee代码仓库 2.下载如下图所示的data.json文件 3.将本地创建的opendb-city-china.schema.json上传到云端 4.点击导入json 如果直接将data.json导入会报错&#xff0c;如下图所示: 5.将data.json本来的数组对象&#…

如何展示科技产品的原理和应用

一、合理安排展示区域 不同的科技产品具有不同的展示需求&#xff0c;设计师需要根据展品的特点和大小&#xff0c;合理安排展示区域。对于较大的科技产品&#xff0c;可以设置特定的展台或展示区域&#xff0c;并配备合适的灯光和装饰&#xff0c;以凸显产品的重要性和独特性。…

InterliJ IDEA基本设置

安装好idea后&#xff0c;将软件打开&#xff0c;可以进行基础设置 1.打开软件&#xff0c;先安装插件-汉化包&#xff08;不推荐&#xff0c;最好使用英文版&#xff09;&#xff0c;本次我们使用汉化版本完成基本设置&#xff0c;后期希望大家适应英文版的开发环境。&#x…

Linux 学习之路 -- 进程篇 2 -- 进程简介

一、简单介绍 在我们写完程序后&#xff0c;由编译器形成二进制的文件&#xff0c;而这个文件在磁盘中存放&#xff0c;磁盘在外设中&#xff0c;如果我们要执行程序&#xff0c;就必需要把程序加载到内存。我们可以打开任务管理器&#xff0c;就可以发现我们的电脑会运行很多…

435.无重叠区间

// 定义一个名为Solution的类 class Solution {// 定义公共方法eraseOverlapIntervals&#xff0c;输入为二维整数数组intervals&#xff0c;返回值类型为整数public int eraseOverlapIntervals(int[][] intervals) {// 首先&#xff0c;按照区间的起始位置对区间进行升序排序A…

【Algorithms 4】算法(第4版)学习笔记 23 - 5.4 正则表达式

文章目录 前言参考目录学习笔记1&#xff1a;正则表达式1.1&#xff1a;表示1.2&#xff1a;快捷表示2&#xff1a;正则表达式与非确定有限状态自动机 REs and NFAs2.1&#xff1a;二元性2.2&#xff1a;模式匹配实现2.3&#xff1a;非确定有限状态自动机 Nondeterministic fin…

慧天[HTWATER]:创新城市水务科技,引领行业变革

【城市内涝水文水动力模型介绍】 慧天[HTWATER]软件&#xff1a;慧天排水数字化分析平台针对城市排水系统基础设施数据管理的需求&#xff0c;以及水文、水力及水质模拟对数据的需求&#xff0c;实现了以数据库方式对相应数据的存储。可以对分流制排水系统及合流制排水系统进行…

PAC下的ROP问题

Armv8.3-A引入了pointer authentication选项。 pointer authentication可以减轻ROP攻击。 为了防止ROP攻击&#xff0c;在函数开始时&#xff0c;LR寄存器中的返回地址被签名。这意味着在寄存器的高位添加了一个PAC。在返回之前&#xff0c;使用PAC对返回地址进行验证。如果检查…

搜索二维矩阵 II - LeetCode 热题 21

大家好&#xff01;我是曾续缘&#x1f497; 今天是《LeetCode 热题 100》系列 发车第 21 天 矩阵第 4 题 ❤️点赞 &#x1f44d; 收藏 ⭐再看&#xff0c;养成习惯 搜索二维矩阵 II 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&…