Event 事件分发和消费机制

原创 2016年08月29日 11:07:51
一、简介
Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev)、onInterceptTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev);能够响应这些方法的控件包括:ViewGroup、View、Activity。方法与控件的对应关系如下表所示:




从这张表中我们可以看到 ViewGroup 和 View 对与 Touch 事件相关的三个方法均能响应,而 Activity 对 onInterceptTouchEvent(MotionEvent ev) 也就是事件拦截不进行响应。另外需要注意的是 View 对 dispatchTouchEvent(MotionEvent ev) 和 onInterceptTouchEvent(MotionEvent ev) 的响应的前提是可以向该 View 中添加子 View,如果当前的 View 已经是一个最小的单元 View(比如 TextView),那么就无法向这个最小 View 中添加子 View,也就无法向子 View 进行事件的分发和拦截,所以它没有 dispatchTouchEvent(MotionEvent ev) 和 onInterceptTouchEvent(MotionEvent ev),只有 onTouchEvent(MotionEvent ev)。

二、Touch 事件分析

事件分发:public boolean dispatchTouchEvent(MotionEvent ev)
Touch 事件发生时 Activity 的 dispatchTouchEvent(MotionEvent ev) 方法会以隧道方式(从根元素依次往下传递直到最内层子元素或在中间某一元素中由于某一条件停止传递)将事件传递给最外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法,并由该 View 的 dispatchTouchEvent(MotionEvent ev) 方法对事件进行分发。dispatchTouchEvent 的事件分发逻辑如下:
如果 return true,事件会分发给当前 View 并由 dispatchTouchEvent 方法进行消费,同时事件会停止向下传递;
如果 return false,事件分发分为两种情况:
如果当前 View 获取的事件直接来自 Activity,则会将事件返回给 Activity 的 onTouchEvent 进行消费;
如果当前 View 获取的事件来自外层父控件,则会将事件返回给父 View 的 onTouchEvent 进行消费。
如果返回系统默认的 super.dispatchTouchEvent(ev),事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。

事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev)
在外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法返回系统默认的 super.dispatchTouchEvent(ev) 情况下,事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。onInterceptTouchEvent 的事件拦截逻辑如下:
如果 onInterceptTouchEvent 返回 true,则表示将事件进行拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理;
如果 onInterceptTouchEvent 返回 false,则表示将事件放行,当前 View 上的事件会被传递到子 View 上,再由子 View 的 dispatchTouchEvent 来开始这个事件的分发;
如果 onInterceptTouchEvent 返回 super.onInterceptTouchEvent(ev),事件默认会被拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理。

事件响应:public boolean onTouchEvent(MotionEvent ev)
在 dispatchTouchEvent 返回 super.dispatchTouchEvent(ev) 并且 onInterceptTouchEvent 返回 true 或返回 super.onInterceptTouchEvent(ev) 的情况下 onTouchEvent 会被调用。onTouchEvent 的事件响应逻辑如下:
如果事件传递到当前 View 的 onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当前 View 向上传递,并且都是由上层 View 的 onTouchEvent 来接收,如果传递到上面的 onTouchEvent 也返回 false,这个事件就会“消失”,而且接收不到下一次事件。
如果返回了 true 则会接收并消费该事件。
如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。
到这里,与 Touch 事件相关的三个方法就分析完毕了。下面的内容会通过各种不同的的测试案例来验证上文中三个方法对事件的处理逻辑。

三、Touch事件测试流程
1)准备:自定义Activity,自定义ViewGroup,自定义View
2)测试

a)全部采用系统默认的方式(super...)
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventChild﹕ 事件处理-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起

b)Activity的事件分发返回true或者false,【结果都一样,只有Activity的事件分发处理】
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件分发-->抬起

c) Activity的事件处理返回true或者false,【结果同上a)一样,系统默认处理情况】
-----------------------------------------------------------------------------------------------------------
a)ViewGroup事件分发返回true
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件分发-->抬起

b)ViewGroup事件分发返回false
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起

c)ViewGroup事件处理返回true
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventChild﹕ 事件处理-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件处理-->抬起

d)ViewGroup事件处理返回false【系统默认处理情况】

e)ViewGroup事件拦截返回true
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起

f)ViewGroup事件拦截返回false
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventChild﹕ 事件处理-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起
-----------------------------------------------------------------------------------------
a)View事件分发返回true
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件拦截-->移动
I/TouchEventChild﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件拦截-->抬起
I/TouchEventChild﹕ 事件分发-->抬起
b)View事件分发返回false
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起
c)View事件处理返回true
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventChild﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件分发-->移动
I/TouchEventFather﹕ 事件拦截-->移动
I/TouchEventChild﹕ 事件分发-->移动
I/TouchEventChild﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件分发-->抬起
I/TouchEventFather﹕ 事件拦截-->抬起
I/TouchEventChild﹕ 事件分发-->抬起
I/TouchEventChild﹕ 事件处理-->抬起
d)View事件处理返回false
I/TouchEventActivity﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件分发-->按下
I/TouchEventFather﹕ 事件拦截-->按下
I/TouchEventChild﹕ 事件分发-->按下
I/TouchEventChild﹕ 事件处理-->按下
I/TouchEventFather﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件处理-->按下
I/TouchEventActivity﹕ 事件分发-->移动
I/TouchEventActivity﹕ 事件处理-->移动
I/TouchEventActivity﹕ 事件分发-->抬起
I/TouchEventActivity﹕ 事件处理-->抬起
版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

研发人员为什么留不住?(2)——原因的解析(上)

上节摘要研发人员管理成了大家都要做但都很痛苦的事,招聘难、考核难、留人难、没赢家!问题产生的原因导致以上现象发生的原因不外乎以下几个:人的原因——永远不要考验人性可以肯定地说,所有项目和公司中产生的问...
  • phphot
  • phphot
  • 2009-06-23 13:24
  • 2105

计算机网络1--基础介绍

本页内容 1.计算机网络定义 2.通信系统模型 3.通信链路 4.交换网络 5.ISP 1.计算机网络定义     定义:计算机网络就是互连的、自治的计算机集合。                (互连...

CSDN博客等级规则

CSDN博客等级规则

Java之求逆矩阵

public class MatrixInverse { public static double Det(double [][]Matrix,int N)//计算n阶行列式(N=n-1) { ...

Servlet简介及其生命周期详解

Servlet简介及其生命周期详解简介        Servlet生命周期,即阐述Servlet从产生到毁灭的整个过程。         ...

【一步步学OpenGL 16】 -《纹理贴图》

教程16纹理贴图基础原文: http://ogldev.atspace.co.uk/www/tutorial16/tutorial16.htmlCSDN完整版专栏: http://blog.csdn....

jscpd--前端代码重复率检测

在实际开发中,一般的重复代码有一下几类 完全一致的代码或者只修改了空格和评论 结构上和句法上一致的代码,例如只是修改了变量名 插入和删除了部分代码 功能和逻辑上一致的代码,语义上的拷贝 很明显越往后,...

C/C++定义全局变量/常量几种方法的区别

C/C++定义全局变量/常量几种方法的区别
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)