安卓触摸事件分发机制分析

1. 前言

🎯 一句话总结:

触摸事件(TouchEvent)会从 Activity 层开始,按从外到内的方式传递给每一个 ViewGroup/View,直到某个 View 消费(consume) 它,事件传递就会停止。

📌 事件分发三个关键方法

方法名所在类作用说明
dispatchTouchEvent()所有 View/ViewGroup事件分发入口,决定是否继续向下传递
onInterceptTouchEvent()仅 ViewGroup是否拦截事件,阻止传递给子 View
onTouchEvent()所有 View/ViewGroup事件的最终处理者(消费者)


DecorView是一个应用窗口的根容器,它本质上是一个FrameLayoutDecorView有唯一一个子View,它是一个垂直LinearLayout,包含两个子元素,一个是TitleView(ActionBar的容器),另一个是ContentView(窗口内容的容器)。

Activity.dispatchTouchEvent()Window.superDispatchTouchEvent()DecorView.dispatchTouchEvent()ViewGroup.dispatchTouchEvent()- onInterceptTouchEvent() → 是否拦截?↓            ↓拦截自己处理     不拦截继续往下↓子View.dispatchTouchEvent()View.dispatchTouchEvent()- onTouchListener.onTouch()- onTouchEvent()
  • Activity.dispatchTouchEvent()
    • 触摸事件从系统层传入,Activity 先接收。
    • 通常会把事件传给当前的 DecorView(根 View)。
  1. ViewGroup.dispatchTouchEvent()
    • 尝试调用 onInterceptTouchEvent() 判断是否拦截事件。
      • 返回 true:表示当前 ViewGroup 要处理,子 View 不再收到事件。
      • 返回 false:继续把事件传给子 View。
  2. View.dispatchTouchEvent()
    如果是 ViewGroup,会重复上面的流程(递归)。
    • 如果是普通 View,直接调用 onTouchEvent()
  3. onTouchEvent()
    • 如果返回 true,表示事件被消费(消费后不会再向上传递)。
    • 如果返回 false,当前控件不处理,事件会被传回上层 ViewGroup 的 onTouchEvent()

2. Activity、ViewGroup、View事件分发机制分析

Activity事件分发机制
  • Activity.dispatchTouchEvent(MotionEvent event)

源码(Activity.java仅关键代码):

public boolean dispatchTouchEvent(MotionEvent ev) {if (ev.getAction() == MotionEvent.ACTION_DOWN) {onUserInteraction();}if (getWindow().superDispatchTouchEvent(ev)) {return true;}return onTouchEvent(ev);
}
  • onUserInteraction():通知用户交互,和事件分发无关。
  • 调用 getWindow().superDispatchTouchEvent(ev)
    • WindowPhoneWindow
    • PhoneWindow 把事件交给了 DecorView(一个 ViewGroup)处理。
  • 如果 superDispatchTouchEvent(ev) 返回 true,说明事件被下面消费了。
  • 否则调用 Activity.onTouchEvent(ev)(比如点击空白处)。
ViewGroup 事件分发机制

DecorViewViewGroup,所以它遵循 ViewGroup 的事件分发规则。

  • ViewGroup.dispatchTouchEvent(MotionEvent ev)

源码(ViewGroup.java仅关键代码):

ViewGroup.dispatchTouchEvent 代码可大致简化为下面这个样子

public boolean dispatchTouchEvent(MotionEvent ev) {// 1. 是否拦截事件boolean intercepted = onInterceptTouchEvent(ev);// 2. 如果没有拦截,遍历子 View 分发if (!intercepted) {for (int i = childrenCount - 1; i >= 0; i--) {final View child = children[i];if (child.dispatchTouchEvent(ev)) {return true;}}}// 3. 子 View 没有消费,自己处理return super.dispatchTouchEvent(ev);
}

详细代码可以查看ViewGroup.dispatchTouchEventdispatchTransformedTouchEvent方法

    private boolean dispatchTransformedTouchEvent(MotionEvent event, boolean cancel,View child, int desiredPointerIdBits) {final boolean handled;// Canceling motions is a special case.  We don't need to perform any transformations// or filtering.  The important part is the action, not the contents.final int oldAction = event.getAction();if (cancel || oldAction == MotionEvent.ACTION_CANCEL) {event.setAction(MotionEvent.ACTION_CANCEL);if (child == null) {handled = super.dispatchTouchEvent(event);} else {handled = child.dispatchTouchEvent(event);}event.setAction(oldAction);return handled;}// 省略....return handled;}
  • onInterceptTouchEvent(ev):决定是否拦截事件(默认返回 false)。
  • 如果返回 true,自己处理,不再传递给子 View。
  • 如果没有拦截,会遍历子 View,调用子 View 的 dispatchTouchEvent(ev)
  • 如果有任何一个子 View 返回了 true,说明消费了事件,整个流程结束。
  • 如果子 View 都没有消费,最后调用自己的 super.dispatchTouchEvent(ev),即作为普通 View 处理。
View 事件分发机制

View 是最终事件的接收者。

  • View.dispatchTouchEvent(MotionEvent ev)

源码(View.java仅关键代码):

public boolean dispatchTouchEvent(MotionEvent event) {boolean result = false;// 1. 先判断是否需要触发 OnTouchListenerListenerInfo li = mListenerInfo;if (li != null && li.mOnTouchListener != null&& (mViewFlags & ENABLED_MASK) == ENABLED&& li.mOnTouchListener.onTouch(this, event)) {result = true;}// 2. 如果 OnTouchListener 没消费,再走 onTouchEventif (!result && onTouchEvent(event)) {result = true;}return result;
}
  • 优先执行 onTouchListener.onTouch()
    • 如果返回 true,表示事件被消费,不继续往下传递。
  • 否则走 onTouchEvent(event)

因为点击事件是在 onTouchEvent 中的 case MotionEvent.ACTION_UP: 中判断调用的,具体查看 View.performClickInternal()方法。

这里就是,如果你设置了某个 ViewOnTouchListener 并且在 onTouch 方法中返回 true,那么这个 ViewonClick 方法不会执行的原因。


3. 理解 ViewGroup 的递归式事件分发?

核心理解:递归式分发

  • 父 ViewGroup 收到事件,先问自己:“要不要拦截?”(onInterceptTouchEvent)**
  • 如果 不拦截,就 找出被点击的子 View
  • 然后 把事件递给子 ViewdispatchTouchEvent() 方法
  • 子 View 又可以是一个 ViewGroup(比如 LinearLayout),于是子 View 又重复上面的流程: onInterceptTouchEvent()
  • 再分发给自己的子 View。
  • 就这样,一层一层递归下去,直到遇到一个普通 View(没有子 View 的 Button、TextView),最后交给 onTouchEvent() 来消费。

打个通俗比喻:

  • 一个 ViewGroup 就像一个"村长",负责分发任务。
  • 它收到任务(MotionEvent)后,会问:
    • 我要自己干?(拦截)
    • 还是派给手下某个小村民?(子 View)
  • 村民又是个小村长(嵌套 ViewGroup)的话,继续往下派。
  • 最后一个真正干活的是普通农民(Button/TextView)。

补充个知识点:

onInterceptTouchEvent() 只在 “ACTION_DOWN” 开始时有意义!!

  • 因为一旦一个手指 ACTION_DOWN 被拦截了,后续的 ACTION_MOVE / ACTION_UP 事件都跟着这个处理链走。
  • 如果 DOWN 没拦截,后面的 MOVE/UP 也不会随便切换到拦截。

这叫做 事件的捕获(capture)机制,Android 保证事件流动的一致性。


4. 最后

Android 事件分发是一层层向下传,遇到拦截或者消费就停;否则事件会向上传递,直到有人消费或者丢弃。

还有一个问题,对于没了解过事件分发机制的同学来说,对于事件分发:由外到内,事件消费:由内到外 的理解,可能有些困惑,包括我自己,其实就可以简单理解为 ViewGroup 一个方法把事件分发机制写完,方法中间就是挨个遍历子 View 挨个问,你要不要这个事件(由外到内),都不要的话,我就要了(又回到了 由内到外 ),带着这个理解,去看源码就很容易理解事件分发机制了。

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

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

相关文章

Spring MVC 多个拦截器的执行顺序

一、流程总览 该流程图描述了一个多层拦截器链的业务处理流程,核心逻辑为: 前置拦截:通过 predHandler1 和 predHandler2 逐层校验请求合法性。核心处理:通过校验后执行核心业务逻辑 handler()。后置处理与清理:按反…

django filter 排除字段

在Django中,当你使用filter查询集(QuerySet)时,通常你会根据模型的字段来过滤数据。但是,有时你可能想要排除某些特定的字段,而不是过滤这些字段。这里有几种方法可以实现这一点: 使用exclude方…

ByeCode,AI无代码开发平台,拖拽式操作构建应用

ByeCode是什么 ByeCode 是一款先进的 AI 无代码平台,旨在帮助企业迅速创建数字名片、网站、小程序、应用程序及内部管理系统,无需繁杂的编码或开发工作。ByeCode 采用直观的可视化界面和拖拽式操作,使得非技术用户能够轻松上手。同时&#x…

AI日报 - 2025年04月28日

🌟 今日概览(60秒速览) ▎🤖 能力进展 | Gemini 2.5 Pro成功挑战《口袋妖怪红》8道馆;AI推理器具备自我纠错能力;LLM在游戏、多模态理解、代码迁移等方面展现新能力。 ▎💼 商业动向 | Google回应DOJ反垄断案&#xff…

在Java中实现List按自定义顺序排序的几种方案

在Java中实现List按自定义顺序排序的几种方案 在实际开发中&#xff0c;我们经常需要对集合中的对象按照特定字段进行排序。当排序规则不是简单的字母或数字顺序&#xff0c;而是自定义的顺序时&#xff0c;我们需要采用特殊的方法。本文将以一个List<Person>按省份特定…

微服务架构在云原生后端的深度融合与实践路径

📝个人主页🌹:一ge科研小菜鸡-CSDN博客 🌹🌹期待您的关注 🌹🌹 一、引言:后端架构的演变,走向云原生与微服务融合 过去十余年,后端架构经历了从单体应用(Monolithic)、垂直切分(Modularization)、到微服务(Microservices)的演进,每一次变化都是为了解决…

Python中的Walrus运算符分析

Python中的Walrus运算符&#xff08;:&#xff09;是Python 3.8引入的一个新特性&#xff0c;允许在表达式中同时赋值和返回值。它的核心作用是减少重复计算&#xff0c;提升代码简洁性。以下是其适用的典型场景及示例&#xff1a; 1. 在循环中避免重复计算 当循环条件需要多次…

用Node.js施展文档比对魔法:轻松实现Word文档差异比较小工具,实现Word差异高亮标注(附完整实战代码)

引言&#xff1a;当「找不同」遇上程序员的智慧 你是否经历过这样的场景&#xff1f; 法务同事发来合同第8版修改版&#xff0c;却说不清改了哪里 导师在论文修改稿里标注了十几处调整&#xff0c;需要逐一核对 团队协作文档频繁更新&#xff0c;版本差异让人眼花缭乱 传统…

前端浏览器窗口交互完全指南:从基础操作到高级控制

浏览器窗口交互是前端开发中构建复杂Web应用的核心能力&#xff0c;本文深入探讨23种关键交互技术&#xff0c;涵盖从传统API到最新的W3C提案&#xff0c;助您掌握跨窗口、跨标签页的完整控制方案。 一、基础窗口操作体系 1.1 窗口创建与控制 // 新窗口创建&#xff08;现代浏…

Git和Gitlab的部署和操作

一。GIT的基本操作 1.GIT的操作和查看内容 [rootmaster ~]# yum install git -y [rootmaster ~]# git config --list&#xff1a;查看所有配置 2.GIT仓库初始化 [rootmaster ~]# mkdir /gittest&#xff1a;创建目录 [rootmaster ~]# cd /gittest/&#xff1a;进入目录 [rootm…

Linux中线程池的简单实现 -- 线程安全的日志模块,策略模式,线程池的封装设计,单例模式,饿汉式单例模式,懒汉式单例模式

目录 1. 对线程池的理解 1.1 基本概念 1.2 工作原理 1.3 线程池的优点 2. 日志与策略模式 2.1 日志认识 2.2 策略模式 2.2.1 策略模式的概念 2.2.2 工作原理 2.2 自定义日志系统的实现 3. 线程池设计 3.1 简单线程池的设计 3.2 线程安全的单例模式线程池的设计 3…

量子力学:量子通信

量子通信是利用量子力学原理对信息进行编码、传输和处理的新型通信方式&#xff0c;以下是其详细介绍及业界发展现状&#xff1a; 基本原理 量子叠加态 &#xff1a;量子系统可以处于多个状态的叠加&#xff0c;如光子的偏振方向可以同时处于水平和垂直方向的叠加态&#xff…

企业架构之旅(1):TOGAF 基础入门

大家好&#xff0c;我是沛哥儿。今天我们简单聊下TOGAF哈。 文章目录 一、TOGAF 是什么定义与核心定位发展历程与行业地位与其他架构框架的区别 二、TOGAF 核心价值企业数字化转型助力业务与 IT 的协同作用降本增效与风险管控 三、TOGAF 基础术语解析架构域&#xff08;业务、…

CSS 内容超出显示省略号

CSS 内容超出显示省略号 文章目录 CSS 内容超出显示省略号**1. 单行文本省略&#xff08;常用&#xff09;****2. 多行文本省略&#xff08;如 2 行&#xff09;****3. 对非块级元素生效****完整示例****注意事项** 在 CSS 中实现内容超出显示省略号&#xff0c;主要通过控制文…

路由器重分发(OSPF+RIP),RIP充当翻译官,OSPF充当翻译官

路由器重分发&#xff08;OSPFRIP&#xff09; 版本 1 RIP充当翻译官 OSPF路由器只会OSPF语言&#xff1b;RIP路由器充当翻译官就要会OSPF语言和RIP语言&#xff1b;则在RIP中还需要将OSPF翻译成RIPOSPF 把RIP路由器当成翻译官&#xff0c;OSPF路由器就只需要宣告自己的ip&am…

AlexNet网络搭建

AlexNet网络模型搭建 环境准备 首先在某个盘符下创建一个文件夹&#xff0c;就叫AlexNet吧&#xff0c;用来存放源代码。 然后新建一个python文件&#xff0c;就叫plot.py吧&#xff0c;往里面写入以下代码&#xff0c;用于下载数据集&#xff1a; # FashionMNIST里面包含了…

【计算机网络】网络基础概念

&#x1f4da; 博主的专栏 &#x1f427; Linux | &#x1f5a5;️ C | &#x1f4ca; 数据结构 | &#x1f4a1;C 算法 | &#x1f152; C 语言 | &#x1f310; 计算机网络 这是博主计算机网络的第一篇文章&#xff0c;本文由于是基础概念了解&#xff0c;引用了大…

在Spring Boot项目中实现Word转PDF并预览

在Spring Boot项目中实现Word转PDF并进行前端网页预览&#xff0c;你可以使用Apache POI来读取Word文件&#xff0c;iText或Apache PDFBox来生成PDF文件&#xff0c;然后通过Spring Boot控制器提供文件下载或预览链接。以下是一个示例实现步骤和代码&#xff1a; 1. 添加依赖 …

图解 Redis 事务 ACID特性 |源码解析|EXEC、WATCH、QUEUE

写在前面 Redis 通过 MULTI、EXEC、WATCH 等命令来实现事务功能。Redis的事务是将多个命令请求打包&#xff0c;然后一次性、按照顺序的执行多个命令的机制&#xff0c;并且在事务执行期间&#xff0c;服务器不会中断事务而该去执行其他客户端的命令请求。 就像下面这样&#…

LeetCode --- 446 周赛

题目列表 3522. 执行指令后的得分 3523. 非递减数组的最大长度 3524. 求出数组的 X 值 I 3525. 求出数组的 X 值 II 一、执行指令后的得分 照着题目要求进行模拟即可&#xff0c;代码如下 // C class Solution { public:long long calculateScore(vector<string>&…