漫画Android:事件分发的过程是怎样的?

article/2025/8/7 11:43:52

1
2
3
4
5
当用户触摸屏幕时,硬件层会捕获触摸信号,并将其转化为内核事件。
Android系统会通过InputManagerService和WindowManagerService等服务将这些事件包装成MotionEvent对象,并将其传递给Activity的dispatchTouchEvent()方法中,Activity会先将事件分发给Window处理,Window调用superDispatchTouchEvent()方法,将事件交给 PhoneWindow处理,然后 PhoneWindow将事件传递给当前窗口的根视图(通常是DecorView,一个FrameLayout)。
DecorView是PhoneWindow的顶级视图,它是所有应用UI的容器。从这里开始,事件分发就进入了应用程序的视图层级

即,事件收集之后最先传递给Activity,随后依次向下传递:

Activity ——> Window ——> …… ——> DecorView ——> ViewGroup ——> …… ——> View

事件分发的流程

整个事件分发过程主要围绕MotionEvent对象展开,并且涉及三个关键方法:dispatchTouchEvent()onInterceptTouchEvent()onTouchEvent()

  1. 事件产生:用户触摸屏幕,系统生成MotionEvent
  2. 根视图分发MotionEvent首先传递给当前Activity,然后依次传递到当前窗口的根视图(DecorView)。
  3. dispatchTouchEvent()DecorView调用自己的dispatchTouchEvent(),决定是否将事件向下分发。
  4. onInterceptTouchEvent() (ViewGroup)
    • 如果DecorViewViewGroup(通常是),它会调用onInterceptTouchEvent()来判断是否拦截事件。
    • 如果返回true(拦截),事件将直接传递给DecorViewonTouchEvent()
    • 如果返回false(不拦截),事件将继续向下分发给子View。
  5. 向下分发DecorView遍历其子View,找到触摸区域内的子View,并调用该子View的dispatchTouchEvent()。这个过程会递归地重复步骤4和5,直到事件到达最底层的View或者被某个ViewGroup拦截。
  6. onTouchEvent() (View/ViewGroup)
    • 如果事件被某个View或ViewGroup拦截(onInterceptTouchEvent()返回true),或者事件一直分发到了最底层的View且没有被任何父ViewGroup拦截,那么该View/ViewGroup的onTouchEvent()方法将被调用。
    • 如果onTouchEvent()返回true,表示该View/ViewGroup消费了事件,事件传递流程结束。
    • 如果onTouchEvent()返回false,表示该View/ViewGroup不处理事件,事件会回溯到其父ViewGroup的onTouchEvent()方法(如果父ViewGroup之前没有拦截该事件)。
  7. 事件未被处理:如果事件最终没有被任何View或ViewGroup处理(即所有onTouchEvent()都返回false),那么该事件会沿着View树向上回溯,最终可能会被Activity的onTouchEvent()方法处理。如果连Activity的onTouchEvent()也返回false,则该事件将被丢弃。

Android事件分发是一个自上而下分发、自下而上处理(如果未被处理)的过程。

6
7
8
9

ViewGroup事件分发流程

依葫芦画瓢!ViewGroup的事件分发流程围绕刚刚提到的三个核心方法展开:dispatchTouchEvent()onInterceptTouchEvent()onTouchEvent()

  1. 事件到达:当一个MotionEvent到达ViewGroup时,首先调用其dispatchTouchEvent()方法。
  2. 是否拦截?:在dispatchTouchEvent()内部,首先调用onInterceptTouchEvent()来判断ViewGroup是否要拦截这个事件。
    • 如果onInterceptTouchEvent()返回true (拦截)
      • 事件不再向下分发给子View
      • ViewGroup自身的onTouchEvent()方法会被调用,以处理该事件。
      • 如果onTouchEvent()返回true,表示事件被ViewGroup消费。
      • 如果onTouchEvent()返回false,表示事件未被ViewGroup消费,事件会回溯到其父ViewGrouponTouchEvent()(如果父ViewGroup之前没有拦截该事件)。
    • 如果onInterceptTouchEvent()返回false (不拦截)
      • ViewGroup会遍历其子View
      • 它会判断触摸点是否在某个子View的范围内。
      • 如果找到合适的子View,就调用该子ViewdispatchTouchEvent()方法,将事件继续向下传递。
      • 如果子ViewdispatchTouchEvent()返回true (子View消费了事件):整个事件分发流程结束。
      • 如果子ViewdispatchTouchEvent()返回false (子View未消费事件)ViewGroup会继续尝试将事件分发给下一个合适的子View
      • 如果所有子View都遍历完了,但都没有消费事件,或者根本没有子View:那么事件会回传给当前ViewGroup,调用其onTouchEvent()方法来处理。
        • 如果ViewGrouponTouchEvent()返回true,事件被ViewGroup消费。
        • 如果ViewGrouponTouchEvent()返回false,事件未被ViewGroup消费,会继续回溯到父ViewGroup

关键点:

  • 优先拦截onInterceptTouchEvent()优先于子ViewdispatchTouchEvent()被调用,它有“一票否决权”。
  • 消费即止:一旦某个ViewViewGrouponTouchEvent()返回true,表示它消费了该事件序列,后续事件将直接传递给它,不再进行分发。
  • 回溯机制:如果一个事件沿着分发路径一直没有被消费(所有onTouchEvent()都返回false),它会沿着调用链向上回溯,最终可能由ActivityonTouchEvent()处理。

内部逻辑(伪代码表示):

public boolean dispatchTouchEvent(MotionEvent event) {boolean handled = false; // 标记事件是否被处理// 1. 调用 onInterceptTouchEvent() 判断是否拦截if (onInterceptTouchEvent(event)) {// 2. 如果 onInterceptTouchEvent() 返回 true (拦截)//    则事件不再向下分发给子View,而是直接交给当前ViewGroup的 onTouchEvent() 处理handled = onTouchEvent(event);} else {// 3. 如果 onInterceptTouchEvent() 返回 false (不拦截)//    则遍历子View,尝试将事件分发给它们// 查找触摸点所在的子View,并调用其 dispatchTouchEvent()// 伪代码:// for (int i = 0; i < getChildCount(); i++) {//     View child = getChildAt(i);//     if (child.isTouchedInBounds(event)) { // 检查触摸点是否在子View范围内//         if (child.dispatchTouchEvent(event)) { // 递归调用子View的dispatchTouchEvent()//             handled = true; // 子View消费了事件//             break; // 停止遍历,事件已被处理//         }//     }// }// 4. 如果所有子View都没有消费事件 (handled 仍为 false)//    或者根本没有子View//    则将事件交给当前ViewGroup的 onTouchEvent() 处理if (!handled) {handled = onTouchEvent(event);}}return handled; // 返回事件是否被处理
}

10


http://www.hkcw.cn/article/cuuHZISaeM.shtml

相关文章

【算法提升】分组 day_tow

1.分组 1.1 解析 个人认为这题最难的点在于如何想到使用二分的算法来解题。 正向求解&#xff1a;就是去看每一组中需要分多少个人&#xff0c;但是这样求解代码我根本写不出来。 所以根据正难则反的思想&#xff0c;我们可以从最终结果去倒推。 枚举最终的分配结果中&#xff…

【笔记】Suna 部署之 Supabase 数据库 schema 暴露操作

#工作记录 一、前置信息 在 Suna 部署过程中&#xff0c;Supabase 数据库设置已完成&#xff08;✅ Supabase database setup completed &#xff09;&#xff0c;但需要手动在 Supabase 平台暴露basejump模式&#xff08;schema&#xff09;。 Suna 部署过程中&#xff0c;S…

【Linux 学习计划】-- 进程状态 | 进程运行、阻塞和挂起的本质 | 并行、并发与进程切换 | 进程优先级

目录 进程状态 五状态进程模型 运行、就绪状态的本质 阻塞状态的本质 挂起状态 并行与并发 进程切换 进程优先级 结语 进程状态 进程状态的本质是什么&#xff1f; 首先我们知道&#xff0c;在操作系统中&#xff0c;进程是需要被管理起来的&#xff0c;具体则是用一…

自证式推理训练:大模型告别第三方打分的新纪元

1. 传统验证体系的困境与技术跃迁的必然性 1.1 传统验证器的局限性 现有强化学习框架依赖显式验证器对答案进行二值化判定&#xff0c;这种模式在数学、代码等可验证领域表现优异。某厂内部数据显示&#xff0c;传统R1-Zero方法在代码生成任务中准确率达92%&#xff0c;但切换…

《操作系统真相还原》——加载器

显存 将上一章的中断输出&#xff0c;变为显存输出 加载器 使用mbr引导程序从磁盘中加载loader程序。 MBR %include "boot.inc" SECTION MBR vstart0x7c00 mov ax,cs mov ds,axmov es,axmov ss,axmov fs,axmov sp,0x7c00mov ax,0xb800mov gs,ax;cl…

Spring Boot 应用中实现配置文件敏感信息加密解密方案

Spring Boot 应用中实现配置文件敏感信息加密解密方案 背景与挑战 &#x1f6a9;一、设计目标 &#x1f3af;二、整体启动流程 &#x1f504;三、方案实现详解 ⚙️3.1 配置解密入口&#xff1a;EnvironmentPostProcessor3.2 通用解密工具类&#xff1a;EncryptionTool 四、快速…

前端实现图片压缩:基于 HTML5 File API 与 Canvas 的完整方案

在 Web 开发中,处理用户上传的图片时,前端压缩可以有效减少服务器压力并提升上传效率。本文将详细讲解如何通过<input type="file">实现图片上传,结合 Canvas 实现图片压缩,并实时展示压缩前后的图片预览和文件大小对比。 一、核心功能架构 我们将实现以…

用wireshark抓了个TCP通讯的包

昨儿个整理了下怎么用wireshark抓包&#xff0c;链接在这里&#xff1a;捋捋wireshark 今天打算抓个TCP通讯的包试试&#xff0c;整体来说比较有收获&#xff0c;给大家汇报一下。 首先就是如何搞到可以用来演示TCP通讯的客户端、服务端&#xff0c;问了下deepseek&#xff0c;…

AWS WAF设置IP白名单

目标 设置一个组白名单IP地址&#xff0c;当发现是这些IP地址发过来的请求后&#xff0c;WAF自动放行。 创建IP集 打开WAF页面&#xff0c;开始IP集创建如下图&#xff1a; 设置ip集&#xff0c;如下图&#xff1a; aws waf acl配置白名单 找到Web ACL&#xff0c;开始在…

随笔20250530 C# 整合 IC卡读写技术解析与实现

以下是一个完整、最简化的 FeliCa 读取整合示例&#xff08;无需 SDK&#xff0c;基于 PCSC NuGet 包&#xff09;&#xff0c;你可以直接运行这个控制台程序&#xff0c;验证能否识别 RC-S300 并读取卡片 UID&#xff1a; &#x1f9ea; 示例说明 &#x1f4e6; 使用 NuGet 包…

day024-网络基础-TCP与UDP、DNS

文章目录 1. 李导推荐书籍2. OSI七层模型2.1 传输层2.2 网络层2.2.1 问&#xff1a;两端处于不同局域网的设备怎么网络通信&#xff1f; 2.3 数据链路层2.4 物理层2.5 图解OSI七层模型 3. 数据传输模式3.1 全双工3.2 半双工3.3 单工 4. TCP 3次握手4.1 抓包 5. TCP 4次挥手5.1 …

AI赋能开源:如何借助MCP快速解锁开源项目并提交你的首个PR

引子 很多同学都梦想为开源项目贡献力量&#xff0c;然而现实往往是——面对庞大复杂的项目&#xff0c;从入门到提交第一个有实质性代码的PR&#xff0c;时间跨度可能长达数年。传统路径通常是先从文档贡献开始&#xff0c;逐步深入理解项目架构&#xff0c;最终才能进行代码…

智能问数技术路径对比:NL2SQL vs NL2Semantic2SQL

在人工智能浪潮席卷数据分析领域的当下&#xff0c;“智能问数”凭借其自然语言交互的便捷性&#xff0c;迅速成为企业提升数据民主化与决策效率的焦点。大语言模型&#xff08;LLM&#xff09;展现出的强大语言理解和生成能力&#xff0c;无疑为这一愿景启动了引擎。 然而&am…

QT-Creator安装教程(windows)

目录 1&#xff0c;下载 1.1 镜像源下载 1.2 运行下载的exe文件 1.2.1 QT5 版本安装 1.2.2 QT6 版本安装 1.2.3 如何在安装完成之后&#xff0c;继续添加扩展包 1&#xff0c;下载 1.1 镜像源下载 地址&#xff1a;Index of /qtproject/ 根据电脑系统选择下载linux、macO…

Warm-Flow发布1.7.3 端午节(设计器流和流程图大升级)

Warm-Flow发布1.7.3 端午节&#xff08;设计器流和流程图大升级&#xff09; 更新内容项目介绍功能思维导图演示地址官网Warm-Flow视频 更新内容 [feat] 新版流程图通过前端渲染[perf] 美化流程设计器ui[feat] 办理人权限处理器&#xff0c;新增办理人转换接口&#xff0c;比如…

分布式锁Redisson使用

redission为我们提供了方便使用redis集群的方法&#xff0c;可以使用它完成锁的建立。 依赖 <dependency><groupId>org.redisson</groupId><artifactId>redisson</artifactId><version>3.36.0</version></dependency>spring引…

Unity UI系统中RectTransform详解

一、基础代码示例 public GameObject node; var rect node.GetComponent<RectTransform>();Debug.Log($"anchoredPosition----{rect.anchoredPosition}"); Debug.Log($"offsetMin.x--{rect.offsetMin}"); Debug.Log($"offsetMax.x--{rect.of…

神经网络(Neural Networks)

设计神经网络的最初动机是编写能够模仿人类大脑学习和思考方式的软件。现今&#xff0c;神经网络也被称为人工神经网络&#xff0c;其工作方式已经与我们所认为的大脑实际工作方式和学习方式大不相同。 研究神经网络的工作始于 20 世纪 50 年代&#xff0c;之后的一段时间它并不…

DeepSeek‑R1-0528 重磅升级:蚂蚁百宝箱免费、无限量调用

DeepSeek‑R1-0528 重磅升级&#xff1a;蚂蚁百宝箱免费、无限量调用 端午假期前一天&#xff0c;DeepSeek‑R1 更新到了 0528 版本&#xff01; 官方说明&#xff1a;0528 版本在深度思考与推理能力方面显著增强——在数学、编程与通用逻辑等多项基准测评中&#xff0c;表现已…

可定制化货代管理系统,适应不同业务模式需求!

在全球化贸易的浪潮下&#xff0c;货运代理行业扮演着至关重要的角色。然而&#xff0c;随着市场竞争的日益激烈&#xff0c;货代企业面临着越来越多的挑战&#xff1a;客户需求多样化、业务流程复杂化、运营成本上升、利润空间压缩……这些挑战迫使货代企业不断寻求创新和突破…