Android触摸事件分发机制(一)

news2025/1/10 14:05:58

1. 简介

本文主要分享事件分发中的基本概念。

介绍负责参与分发事件的主要方法。

从这些方法的核心逻辑中,总结事件分发的规律。

2. 被分发的对象


被分发的对象是那些?被分发的对象是用户触摸屏幕而产生的点击事件,事件主要包括:按
下、滑动、抬起与取消。这些事件被封装成 MotionEvent 对象。该对象中的主要事件如下表
所示:

按下、滑动、抬起、取消这几种事件组成了一个事件流。事件流以按下为开始,中间可能有
若干次滑动,以抬起或取消作为结束。
在安卓对事件分发的处理过程中,主要是对按下事件作分发,进而找到能够处理按下事件的
组件。对于事件流中后续的事件(如滑动、抬起等),则直接分发给能够处理按下事件的组
件。故本文讨论的内容则是主要针对按下事件的。

 3. 分发事件的组件

分发事件的组件,也称为分发事件者,包括 Activity、View 和 ViewGroup。它们三者的一般
结构为:

事件分发者结构
从上图中可以看出,Activity 包括了 ViewGroup,ViewGroup 又可以包含多个 View。 

4. 分发的核心方法 

负责对事件进行分发的方法主要有三个,分别是:
dispatchTouchEvent(
onTouchEvent()
onInterceptTouchEvent()。
它们并不存在于所有负责分发的组件中,其具体情况总结于下面的表格中:

从 表 格 中 看 , dispatchTouchEvent,onTouchEvent 方 法 存 在 于 上 文 的 三 个 组 件 中 。 而
onInterceptTouchEvent 为 ViewGroup 独有。这些方法的具体作用在下文作介绍。
ViewGroup 类中,实际是没有 onTouchEvent 方法的,但是由于 ViewGroup 继承自 View,而
View 拥有 onTouchEvent 方法,故 ViewGroup 的对象也是可以调用 onTouchEvent 方法的。故在表格中表明 ViewGroup 中存在 onTouchEvent 方法的。 

5. 事件分发过程

这一小节是本文的核心内容,会从整体上对事件的分发过程作介绍。
对于事件分发过程从,笔者认为网上的一些教程中的观点是有误的。
网上部分教程认为事件是从内部(如 Button)开始分发的,这是有误的。
网上部分教程常使用’向上‘、’向下‘传播等描述,但又未对‘何为上’、‘何为下’作解释。
网上部分教程将 Java 的子类对象调用父类方法(向上转型)的过程也称为‘向上’传播,即将
事件在组件之间的传播与程序语言多态特性混为一谈,让初学者费解。
子类在覆写的方法中调用父类的同名方法,被称为’向上传播‘,这也是不对的。
为此在介绍分发过程之前,先对一些概念作定义:
向下传播:Activity 包括 Layout,事件从 Activity 向 Layout 传播被称作’向下传播‘。Layout 包
含若干 View,事件从 Layout 向其子 View 传播,也被称为’向下传播‘。
向上传播:与’向下传播‘相反。
’向上转型‘不能称为传播,即子类对象调用父类方法,或在覆写的方法中调用父类方法,都
不能称为传播。不能将面向对象程序语言中的概念与布局层次中的上下传播混为一谈。
分发方法 dispatchTouchEvent
从方法的名称中可以看出该方法主要是负责分发,是安卓事件分发过程中的核心。事件是如
何传递的,主要就是看该方法,理解了这个方法,也就理解了安卓事件分发机制。
在了解该方法的核心机制之前,需要知道一个结论:
如果某个组件的该方法返回 TRUE,则表示该组件已经对事件进行了处理,不用继续调用其余
组件的分发方法,即停止分发。
如果某个组件的该方法返回 FALSE,则表示该组件不能对该事件进行处理,需要按照规则继续
分发事件。在不复写该方法的情况下,除了一些特殊的组件,其余组件都是默认返回 False
的。后续有例子说明。
为何返回 TRUE 就不用继续分发,而返回 FALSE 就停止分发呢?为了解决这个疑问,需要看
一看该方法的具体分发逻辑。为了便于理解,下面对 dispatchTouchEvent 方法进行简化,只
保留最核心的逻辑。
Activity 的 dispatchTouchEvent 方法

// Activity 中该方法的核心部分伪代码
public boolean dispatchTouchEvent(MotionEvent ev) {
if (child.dispatchTouchEvent(ev)) {
    return true; //如果子 View 消费了该事件,则返回 TRUE,让调用者知道该事件已
被消费
} else {
    return onTouchEvent(ev); // 如 果 子 View 没 有 消 费 该 事 件 , 则 调 用 自 身 的
    onTouchEvent 尝试处理。
    }
}

首先,从核心逻辑中看出,当事件传递给 Activity 后,它先将事件分发给子 View 处理。
如果经过子 View 层层传递或处理后,该事件被消费了(即返回了 TRUE),则 Activity 的分
发方法也返回 TRUE,同样也表示该事件已经被消费了。
如果经过子 View 层层传递或处理后,该事件没有被消费(即返回了 FALSE),则 Activity 的
分发方法就不会返回 TRUE 了,而是调用 onTouchEvent()去处理,看其实际的处理情况。
如果 onTouchEvent 消费了事件,那依然能返回 TRUE(表示已消费事件),这个 TRUE 作为
dispatchTouchEvent 的返回值,让调用它的对象知道该 Activity 已经消费了事件。
如果 onTouchEvent 没有消费该事件,那就返回 FALSE(表示未消费事件),这个 FALSE 作为
dispatchTouchEvent 的返回值,让调用它的对象知道该 Activity 没有消费事件,需要继续处理。
ViewGroup 的 dispatchTouchEvent 方法

// ViewGroup 中该方法的核心部分伪代码
public boolean dispatchTouchEvent(MotionEvent ev) {
    if (!onInterceptTouchEvent(ev)) {
        return child.dispatchTouchEvent(ev); //不拦截,则传给子 View 进行分发处理
    } else {
        return onTouchEvent(ev); //拦截事件,交由自身对象的 onTouchEvent 方法处理
    }
}

ViewGroup 的该方法与 Activity 的类似,只是新添了一个 onInterceptTouchEvent 方法。当事
件传入时,首先会调用 onInterceptTouchEvent。
如果该方法返回了 FALSE(表示不拦截),则交给子 View 去调用 dispatchTouchEvent()方

如果该方法返回了 TRUE(表示拦截),则直接交给该 ViewGroup 对象的 onTouchEvent(ev)
方法处理,具体是否能处理以 onTouchEvent()的实际情况为准。
实 际 上 , 在 onInterceptTouchEvent 返 回 TURE 表 示 拦 截 时 , 实 际 调 用 的 是
super.dispatchTouchEvent 方法,即 View 的该方法,进而由该方法调用 onTouchEvent.
View 的 dispatchTouchEvent 方法

// View 中该方法的核心部分伪代码
public boolean dispatchTouchEvent(MotionEvent ev) {
//如果该对象的监听成员变量不为空,则会调用其 onTouch 方法,
    if (mOnTouchListener != null && mOnTouchListener.onTouch(this, event)) {
        return true; // 若 onTouch 方 法 返 回 TRUE , 则 表 示 消 费 了 该 事 件 , 则
        dispachtouTouchEvent 返回 TRUE,让其调用者知道该事件已被消费。
    }
    return onTouchEvent(ev); //若监听成员为空或 onTouch 没有消费该事件,则调用对象自身的        onTouchEvent 方法处理。
}

从该方法的核心逻辑中可以看到,事件传递进来后,首先会对 mOnTouchListener 判空,如
果之前 Set 了 Listener,则会调用其 onTouch 方法。
若 onTouch 方法返回 TRUE,则 dispatchTouchEvent 也会返回 TRUE,表示消费该事件。
若 onTouch 方法返回 FALSE,或者 mOnTouchListener 本来就是空,则调用自身的 onTouchEvent()
来处理,是否消费事件,可以由其返回值判断。
实际上,在 View 的 onTouchEvent 方法中,如果设置了 onClickListener 监听对象,则会调用
其 onClick 方法。
在 同 时 设 置 了 onTouchListener 与 onClickListener 对 象 的 情 况 下 , 正 是 由 于 View 的
dispacthTouchEvent 方法会先调用 mOnTouchListener 的 onTouch,才会调用 onTouchEvent 方法,
所以 onTouchListener 对象的 onTouch 方法是优先于 onClickListener 对象的 onClick 方法调用
的。这里只简单描述结论,具体源码请查看本文对应的高级篇内容。
小节:dispatchTouchEvent 方法
回顾上面 Activity、ViewGroup 和 View 中的 dispatchTouchEvent 方法,它们大体都可以分为
两部分,前一部分是交由子 View 的 dispatchTouchEvent 方法或 onTouch 方法进行处理,后一
部分是交给自身的 onTouchEvent 方法处理。这样理解的话,就非常便于记忆了。
为了便于记忆和理解,可以将各组件的 dispatchTouchEvent 方法分为两部分:
子 View 的 dispatchTouchEvent 或 onTouch 方法
自身的 onTouchEvent 方法
这个结构有点类似于递归的过程,就是组件的 dispatchTouchEvent 会自用子组件的同名方法,
子组件一样会调用子子组件的同名方法,直到递归到底,然后在从递归底部返回上层,直到
返回到最上层,整个过程结束。或者在这个过程中,事件传递到某个子 View,该子 View 决
定处理该事件,则事件交给其自身的 onTouchEvent 方法处理,如果 onTouchEvent 方法处理
不了,再交由父组件的同名方法处理,直到向上传递到顶层结束。
于是,就有了很多教程里的 U 型图。

安卓分发事件 U 型图
从 U 型图中可以发现,其实安卓事件分发的主体思路非常简单,即由父组件不断向子组件
分发,若子组件能够处理,则立刻返回。若子组件都不处理,那传递到底层的子组件,再返
回回来。这个过程类似上面说的递归的过程。
这 里 对 这 个 U 型 图 做 一 下 说 明 , 先 看 图 中 左 上 角 , 事 件 传 到 Activity , 首 先 调 用 其
dispatchTouchEvent 方法,其会传递给子 View 处理,该子 View(在图中是 ViewGroup)会调
用其 dispatchTouchEvent 方法,如果该方法被覆写直接返回 TRUE,则立即返回 Activity,表示
已 经 消 费 事 件 。 如 果 该 方 法 没 有 被 覆 写 或 调 用 了 super 的 同 名 方 法 , 则 会 调 用
onInterceptTouchEvent 方法,如果该方法返回 TRUE 拦截事件,则交给自身的 onTouchEvent
处 理 , 如 果 该 方 法 返 回 FALSE 不 拦 截 , 则 继 续 传 给 子 子 View ( 图 中 是 View ) 的
dispatchTouchEvent 方法处理。此时,再看看这个 U 型图,该递归调用已经到底了,若在该
方法中的 onTouchListener 方法不处理,则调用自身的 onTouchEvent 处理。若还是处理不了,
则从递归底部向上返回,依次调用 ViewGroup 的、Activity 的 onTouchEvent 方法。
实 际 上 , 用 这 个 U 型 图 来 描 述 安 卓 的 事 件 分 发 机 制 并 不 一 定 准 确 , 因 为 同 一 对 象 的
dispatchTouchEvent 方法实际是包含了另外几个方法的(Activity 与 View 只包含 onTouchEvent),
但是在这个图中,却是将几个方法分别画在不同的框中。所以通过该 U 型图来理解事件分
发机智是不准确的。但是对于部分读者可能会有所帮助。要准确理解事件调用机制,还是应
该回到上面,查看三个核心方法的核心逻辑,就能够准确理解。
强调说明,安卓事件分发的‘向上’与‘向下‘传播,不要与面向对象程序语言中基类与子类关系,
或子类向上调用父类方法等概念搞混淆。对于安卓事件分发的‘向上’与‘向下‘传播,这里的上
与下,是指在’递归‘调用过程中的上与下(也体现到 U 型图里的上与下)。这个概念,体现到布局中,就是外与内。即这里所说的事件’向下‘传播,等同于在布局上,由外向内传播,
而’向上’传播,等同于在布局上,由内向外传播。
在面向对象程序语言中,对于子类覆盖父类方法,或子类调用父类方法,这些‘上’与‘下’的关
系,在布局层面上并没有跨越布局层次,不要与事件传播的方向概念相混淆。
拦截方法 onInterceptTouchEvent
该方法是 ViewGroup 类对象所独有的,用于对事件进行提前拦截。在一般情况下,该方法是
默认返回 FALSE 的,即不拦截。
如果自定义的 ViewGroup 希望拦截事件,不希望事件继续往子 View 传播,可以覆写该方法,
返回 TRUE,即可阻止向下的传播过程。
实际上,从上面的核心逻辑的伪代码中可以看出,在 ViewGroup 调用 dispatchTouchEvent 后,
肯定会调用该方法,根据该方法的返回值来确定如何处理。若该方法返回 True,则会将事件
拦截掉,就给自身的 onTouchEvent 处理。如果返回 False,则继续传递给 child 执行分发流程。
处理方法 onTouchEvent
该方法主要对事件进行处理,若返回 True 表示已经处理了事件,若返回 False 则表示没有对
事件进行处理,需要继续传递事件。一般情况下,默认为 FALSE。在 View 的 onTouchEvent
方法中,如果设置了 onClickListener 监听对象,则会调用其 onClick 方法。

6. 总结

本文在介绍了事件分发基本概念的基础上,介绍了负责参与事件分发的核心方法,包括
dispatchTouchEvent()、onInterceptTouchEvent 与 onTouchEvent 方法。通过伪代码的形式介绍
了这些方法的核心逻辑,重点分析了在 Activity、ViewGroup 与 View 中的 dispatchTouchEvent
方法。它们三者中的该方法结构类似,都是先调用子 View 的同名方法或者 listener 方法,然
后再调用自身的 onTouchEvent 方法。
这些方法在调用关系中体现了一个类似‘递归’的调用过程,通过 dispatchTouchEvent 将事件
传 递 下 去 , 又 通 过 onTouchEvent 将 事 件 传 递 上 来 。 中 间 的 这 一 过 程 可 以 通 过 让
onInterceptTouchEvent 方法(对于 ViewGroup),或者另外的负责分发的方法返回 TRUE,均
可以提前终止这一类似’递归‘的调用过程,进而让事件的处理符合我们的预期。

 

 

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

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

相关文章

人脸检测实战-insightface

目录 简介 一、InsightFace介绍 二、安装 三、快速体验 四、代码实战 1、人脸检测 2、人脸识别 五、代码及示例图片链接 简介 目前github有非常多的人脸识别开源项目,下面列出几个常用的开源项目: 1、deepface 2、CompreFace 3、face_recogn…

Moonbeam生态支持有哪些?

Moonbeam作为波卡生态中最活跃的底层公链之一,自上线以来就致力于构建生态,并通过XCM和GMP等技术实现了多链部署的愿景。通过举办针对不同主题的黑客松、建立生态增长基金、设计项目孵化计划和提供开发奖励等方式,Moonbeam持续推动链上生态的…

基于规则指导的知识图谱推理协作代理学习(2019)7.27

基于规则指导的知识图谱推理协作代理学习 摘要介绍问题和准备工作问题公式基于符号的方法基于游走的方法 RuleGuider模型架构实体代理策略网络 模型学习奖励设计训练过程 实验实验设置数据集实验结果消融研究人工评估 总结 摘要 基于 行走模型 是通过在提供可解释决策的同时实…

【C++】-多态的经典题目

💖作者:小树苗渴望变成参天大树🎈 🎉作者宣言:认真写好每一篇博客💤 🎊作者gitee:gitee✨ 💞作者专栏:C语言,数据结构初阶,Linux,C 动态规划算法🎄 如 果 你 …

解密C++多态机制:发挥对象的多样性,实现更加智能的程序设计

目录 一.多态1.多态的用处2.多态的实现3.虚函数4.override 和 final5.重载重写与重定义6.虚函数表 一.多态 1.多态的用处 众所周知C语言的三大特性:封装、多态、继承。其中多态就是去完成某个行为,但是会根据不同的对象产生不同的状态,所以…

Windows SMB 共享文件夹 排错指南

1 排错可能 是否系统名称为全英文格式 如果不是则 重命名 根据如下排错可能依次设置 1,在运行里面输入"secpol.msc"来启动本地安全设置,\ 然后选择本地策略–>安全选项 -->网络安全LAN 管理器身份验证级别,\ “安全设置”…

操作系统知识点总结

操作系统知识点总结: 第一章:操作系统概述 1.1操作系统的概念: ​ 操作系统是一种系统软件,与其他系统软件和应用软件不同,它有自己的基本特征。它的四大基本特征也就是并发,共享,虚拟,异步。 1.2操作系统的特征(四大…

前端开发信息套路:信息是如何传值

文章目录 前言信息组件化的难点:组件信息流通信息流通信息流通分类通知直接调用回调函数发布订阅全局缓存使用建议 总结 前言 作为一个写了好几个月的Uniapp小程序的打工狗,我总结了一下在前端中信息是如何传递的,得出了一个比较标准化的信息…

thinkphp6 验证码验证结果失败,可能是session开启位置错了!!!

搞了一下下午&#xff0c;始终提示验证码不正确 然后百度得到的结果都是&#xff1a;开启session&#xff0c;但是我开启了就是管用 <?php // 全局中间件定义文件 return [// 全局请求缓存// \think\middleware\CheckRequestCache::class,// 多语言加载// \think\middle…

【人人都看懂的漫画算法】边打扑克边学插入排序算法,彻底搞懂时间复杂度

博主&#xff1a;爱码叔 个人博客站点&#xff1a; icodebook.com 公众号&#xff1a;漫话软件设计 微博&#xff1a;程序员涛哥 专注于软件设计与架构、技术管理。擅长用通俗易懂的语言讲解技术。对技术管理工作有自己的一定见解。文章会第一时间首发在个站上&#xff0c;欢迎…

如何评判算法好坏?复杂度深度解析

如何评判算法好坏&#xff1f;复杂度深度解析 1. 算法效率1.1 如何衡量一个算法好坏1.2 算法的复杂度 2 时间复杂度2.1 时间复杂度的概念2.1.1 实例 2.2 大O的渐进表示法2.3 常见时间复杂度计算举例 3 空间复杂度4 常见复杂度对比5 结尾 1. 算法效率 1.1 如何衡量一个算法好坏 …

Android Studio import的代码报红提示找不到,但正常编译运行

Android Studio import的代码报红提示找不到&#xff0c;但正常编译运行 解决方案&#xff1a; &#xff08;1&#xff09;关闭退出Android Studio。 &#xff08;2&#xff09;删掉当前工程项目目录下的.idea和.gradle文件夹。 &#xff08;3&#xff09;Invalidate Caches…

uni-app之微信小程序实现‘下载+保存至本地+预览’功能

目录 一、H5如何实现下载功能 二、微信小程序实现下载资源功能方面与H5有很大的不同 三、 微信小程序实现文件&#xff08;doc,pdf等格式&#xff0c;非图片&#xff09;下载&#xff08;下载->保存->预览&#xff09;功能 四、图片预览、保存、转发、收藏&#xff1…

MMdetection框架速成系列 第07部分:数据增强的N种方法

MMdetection框架实现数据增强的N种方法 1 为什么要进行数据增强2 数据增强的常见误区3 常见的六种数据增强方式3.1 随机翻转&#xff08;RandomFlip&#xff09;3.2 随机裁剪&#xff08;RandomCrop&#xff09;3.3 随机比例裁剪并缩放&#xff08;RandomResizedCrop&#xff0…

F12开发者工具的简单应用

目录 elements 元素 1、元素的定位和修改 2、UI自动化应用 console 控制台 sources 源代码 network 网络 1、定位问题 2、接口测试 3、弱网测试 performance 性能 memory 存储 application 应用 recorder 记录器 界面展示如下&#xff08;设置中可以切换中英文&am…

云共享平台:助力企业数字化转型的利器

随着互联网技术的不断发展&#xff0c;云共享平台正在成为企业发展必不可少的工具之一。那么&#xff0c;云共享平台到底为企业带来了哪些助力呢&#xff1f; 1. 提高效率 云共享平台可以极大地提高企业的工作效率。传统的企业数据管理通常都需要人工操作&#xff0c;而且往往…

redis数据库与主从复制

目录 一 基本操作 二 执行流程 三 reids持久化 四 rdb和aof持久化的过程 五 为什么会有内存碎片 六 redis组从复制 一 基本操作 set :存放数据 例如 set 键值 内容 set k kokoko k就是键值 kokoko就是内容 get:获取数据 例如 get k 就会出来 k对应的数据 keys 查询键…

【大数据之Flume】三、Flume进阶之Flume Agent 内部原理和拓扑结构

1 Flume事务 2 Flume Agent 内部原理 重要组件&#xff1a; 1、ChannelSelector&#xff08;选择器&#xff09;   ChannelSelector 的作用就是选出 Event 将要被发往哪个 Channel。   &#xff08;1&#xff09;Replicating ChannelSelector&#xff08;复制或副本&#x…

Python实现HBA混合蝙蝠智能算法优化LightGBM回归模型(LGBMRegressor算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 蝙蝠算法是2010年杨教授基于群体智能提出的启发式搜索算法&#xff0c;是一种搜索全局最优解的有效方法…

「前缀和以及差分数组」

文章目录 1 前缀和数组1.1 题解1.2 Code1.3 结果 2 二维矩阵的前缀和数组2.1 题解2.2 Code2.3 结果 3 差分数组 1 前缀和数组 适用于快速频繁的计算一个索引区间内的元素之和&#xff0c;核心思想就是使用一个前缀和数组&#xff0c;然后使用前缀和数组的两个元素之差&#xf…