Android进阶——Android事件分发机制之dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent
Android事件分发机制可以说是我们Android工程师面试题中的必考题,弄懂它的原理是我们避不开的任务,所以长痛不如短痛,花点时间干掉他,废话不多说,开车啦
Android事件分发机制的发生在View与View之间或者ViewGroup与View之间具有镶嵌的视图上,而且视图上必须为点击可用。当一个点击事件产生后,它的传递过程遵循如下顺序:Activity->Window->View,即事件先传递给Activity,再到Window,再到顶级View,才开始我们的事件分发
Android事件分发机制主要由三个重要的方法共同完成的
- dispatchTouchEvent:用于进行点击事件的分发
- onInterceptTouchEvent:用于进行点击事件的拦截
- onTouchEvent:用于处理点击事件
这里需要注意的是View中是没有onInterceptTouchEvent()方法的
这里以两个ViewGroup嵌套View来演示,下面是演示图
一、MyView
继承View并覆写其三个构造方法,覆写dispatchTouchEvent和onTouchEvent
二、MyViewGroup01和MyViewGroup02
MyViewGroup01和MyViewGroup02是一样的代码,这里以01为例,继承ViewGroup并覆写其三个构造方法,覆写dispatchTouchEvent和onTouchEvent和onInterceptTouchEvent方法
三、MyView和MyViewGroup布局文件
这里以ViewGroup和Group嵌套,由上面可以知道事件最后分配到布局的顶级View,这里的顶级View是MyViewGroup02,然后开始事件的传递
点击MyView(即蓝色部分):先接收事件的是父容器(MyViewGroup02)往下分发,而事件的分发过程中分为两步骤
- 分发过程
- 处理过程
其正常的分发事件结果为
1、dispatchTouchEvent(分发事件)
如果在MyViewGroup01的dispatchTouchEvent方法中返回true,表示需要在MyViewGroup01消费了整个事件,即不会再分发,也不会再处理。dispatchTouchEvent方法中返回true的打印信息
如果在MyViewGroup01的dispatchTouchEvent方法中返回false,表示在MyViewGroup01点击事件在本层不再继续进行分发,并交由上层控件的onTouchEvent方法进行消费。dispatchTouchEvent方法中返回false的打印信息
2、onInterceptTouchEvent(拦截事件)
如果在MyViewGroup01的onInterceptTouchEvent方法中返回true,表示需要在MyViewGroup01拦截这个点击事件,不再继续往下分发,即MyView不再执行dispatchTouchEvent方法。但是只是分发结束了而已,接着开始处理事件。下面是onInterceptTouchEvent方法中返回true的打印信息
如果在MyViewGroup01的onInterceptTouchEvent方法中返回false,表示需要在MyViewGroup01不会拦截这个点击事件,继续往下分发。下面是onInterceptTouchEvent方法中返回false的打印信息
3、onTouchEvent(消费事件)
如果MyViewGroup01的onTouchEvent方法中返回true,表示MyViewGroup01可以将该事件直接消费掉了,即分发结束后,处理事件的时候,直接处理到MyViewGroup01就可以结束了。下面是onTouchEvent方法中返回true的打印信息
如果MyViewGroup01的onTouchEvent方法中返回false,表示MyViewGroup01不可以将该事件直接消费掉,即事件继续往上处理。下面是onTouchEvent方法中返回false的打印信息
总结起来:
- dispatchTouchEvent
- return true:表示该View内部消化掉了所有事件
- return false:表示事件在本层不再继续进行分发,并交由上层控件的onTouchEvent方法进行消费
- return
super.dispatchTouchEvent(ev):默认事件将分发给本层的事件拦截onInterceptTouchEvent方法进行处理
- onInterceptTouchEvent
- return true:表示将事件进行拦截,并将拦截到的事件交由本层控件的onTouchEvent进行处理
- return false:表示不对事件进行拦截,事件得以成功分发到子View
- return
super.onInterceptTouchEvent(ev):默认表示不拦截该事件,并将事件传递给下一层View的dispatchTouchEvent
- onTouchEvent
- return true:表示onTouchEvent处理完事件后消费了此次事件
- return
fasle:表示不响应事件,那么该事件将会不断向上层View的onTouchEvent方法传递,直到某个View的onTouchEvent方法返回true - return super.dispatchTouchEvent(ev):表示不响应事件,结果与return
false一样
这里以网上的图片来说明,如果对上面分发例子还不太懂的同学,看这张图片已经很生动的说明了整个过程
我们这里以ViewGroup的dispatchTouchEvent()方法开始讲解,这里面主要有两件事情
- 询问是否拦截事件
- 遍历子View并分发事件
一、dispatchTouchEvent源码询问拦截事件
ViewGroup在两种情况下都会判断是否要拦截当前事件
- 事件类型为ACTION_DOWN:当前由我们触发的点击事件,也即是说ACTION_MOVE和ACTION_UP事件来时,则不触发拦截事件
- mFirstTouchTarget
!=null:当ViewGroup不拦截事件并将事件交给子View的时候该不等式成立。反过来,事件被ViewGroup拦截时,该不等式不成立
二、dispatchTouchEvent源码遍历子View并分发事件
ViewGroup直接使用for遍历所有子View,对子View的各种状态进行判断,最后调用dispatchTransformedTouchEvent(ev,
false, child,
idBitsToAssign)将事件传递给子View,下面是dispatchTransformedTouchEvent()方法的部分源码
其最后就是分发给子View的dispatchTouchEvent()方法,那么事件就分发到子View中去了
View对点击事件的处理过程主要有
- View的dispatchTouchEvent():判断分发事件
- View的onTouchEvent():处理事件的具体做法
一、dispatchTouchEvent源码判断分发事件部分
从源码判断处看出,首先会判断有没有设置mOnTouchListener,如果mOnTouchListener不为空,那么onTouchEvent就不会被调用,这里可以得到一个结论,若在View中设置了OnTouchListener,那么它的优先级是高于onTouchEvent的,这样可以更好的让我们自己setOnTouchEventListener()处理点击事件
二、onTouchEvent源码处理事件的具体做法部分
从对点击事件的具体处理中看出,只要View的CLICKABLE和LONG_CLICKABLE有一个为true,那么它就会消耗这个事件,即onTouchEvent方法返回true。在ACTION_UP事件中,会触发PerformClick()方法,如果View设置了OnClickListener,那么PerformClick()方法内部会调用它的onClick()方法,这里就不分析它的点击事件了
onTouchListener > onTouchEvent > onLongClickListener >
onClickListener
1、简要的谈谈Android的事件分发机制?
当点击事件发生时,首先Activity将TouchEvent传递给Window,再从Window传递给顶层View。TouchEvent会最先到达最顶层
view 的 dispatchTouchEvent ,然后由 dispatchTouchEvent
方法进行分发,如果dispatchTouchEvent返回true ,则整个事件将会被销毁,如果dispatchTouchEvent返回 false
,则交给上层view的 onTouchEvent 方法来开始处理这个事件,如果 interceptTouchEvent 返回 true
,也就是拦截掉了,则交给自身的 onTouchEvent 来处理,如果 interceptTouchEvent 返回 false ,那么事件将继续传递给子
view ,由子 view 的 dispatchTouchEvent 再来开始这个事件的分发。如果事件传递到某一层的子 view 的 onTouchEvent
上了,且这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 onTouchEvent
来接收,tips直到onTouchEvent返回true为止。而如果传递到最顶view的 onTouchEvent 也返回 false 的话,这个事件就会消失。
2、为什么View有dispatchTouchEvent方法?
因为View可以注册很多事件的监听器,如长按、滑动、点击等,它也需要一个管理者来分发
3、ViewGroup中可能有很多个子View,如何判断应该分配给哪一个?
根据源码可知,它会分配给在点击范围内的子View
4、当点击时,子View重叠应该如何分配?
一般分配给最上层的子View,这是由于安卓的渲染机制导致的
事件分发机制就犹如数学的定理是一样道理的,只有记住定理,才能在具体应用中具体分析,有人可能不知道在什么地方会用到,如果你做的项目中,比如有一个控件点击不能反应,那么就有可能是事件分发的结果。至于对源码的分析可能内容比较复杂,内容也多,源码部分有些也让我不是很懂,这里是浅析一下,做个开头,希望大家下去有时间可以翻源码去理解这一机制,将它运用在实战中吧
Android进阶——Android事件分发机制之dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent的更多相关文章
- Android中的事件分发机制
Android中的事件分发机制 作者:丁明祥 邮箱:2780087178@qq.com 这篇文章这周之内尽量写完 参考资料: Android事件分发机制完全解析,带你从源码的角度彻底理解(上) And ...
- Android view 的事件分发机制
1 事件的传递顺序是 Activity -> Window -> 顶层View touch 事件产生后,最先由 activity 的 dispatchTouchEvent 处理 /** * ...
- Android中的事件分发机制总结
Android 的事件分发机制 一.View的事件分发总结: View的onTouchEvent和OnTouch区别 还是以自定义的TestButton为例. 我们可以通过重写onTouchEven ...
- Android的Touch事件分发机制简单探析
前言 Android中关于触摸事件的分发传递是一个很值得研究的东西.曾不见你引入了一个ListView的滑动功能,ListView就不听你手指的指唤来滚动了:也不知道为啥Button设置了onClic ...
- Android View的事件分发机制
准备了一阵子,一直想写一篇事件分发的文章总结一下.这个知识点实在是太重要了. 一个应用的布局是丰富的,有TextView,ImageView,Button等.这些子View的外层还有ViewGroup ...
- Android View的事件分发机制和滑动冲突解决方案
这篇文章会先讲Android中View的事件分发机制,然后再介绍Android滑动冲突的形成原因并给出解决方案.因水平有限,讲的不会太过深入,只希望各位看了之后对事件分发机制的流程有个大概的概念,并且 ...
- Android程序员事件分发机制学习笔记
通过问题来学习一个东西是很好的方法.学习Android中View的事件体系,我也通过给自己提问题,在解决问题的同时也就知道了其中原理. 首先来几个问题起步: 什么是事件?什么是事件分发机制? 在我们通 ...
- Android View的事件分发机制探索
概述 Android事件传递机制也是Android系统中比较重要的一块,事件类型有很多种,这里主要讨论TouchEvent的事件在framework层的传递处理机制.因为对于App开发人员来说,理解f ...
- 浅谈Android中的事件分发机制
View事件分发机制的本质就是就是MotionEvent事件的分发过程,即MotionEvent产生后是怎样在View之间传递及处理的. 首先介绍一下什么是MotionEvent.所谓MotionEv ...
随机推荐
- Vue中 v-for 绑定key和不绑定key的区别
首先,它们区别主要在于 虚拟DOM的复用,绑定key可以更好的复用,下面来详细讲一下 假如我们有一个数组 arr = [1,2,3,4],我们要在2后面插入一个值9: 如果绑定了key值,那么会是这样 ...
- 个人项目发表到GitHub
个人项目源程序代码发送到github上. 因为之前假期已经注册过了账户,所以只是上传了做过的小账本项目. https://www.cnblogs.com/sdcs/p/8270029.html
- python编写banner获取的常用模块
模块的概念:模块也叫库,每个模块中都内置了大量的功能和函数.类和变量.它就像是积木,可以根据需要进行调用组合.模块就是程序,每个模块就是一个后缀为.py的Python程序.Python的模块分为标准模 ...
- HTML学习第七天(一)
HTML学习第七天(一) aside元素用来表示当前或文章的附属信息部分,它可以包含与当前页面或主要内容的相关引用.侧边栏.广告.导航条,以及其他类似的有区别于主要内容的部分 <!DOCTYPE ...
- PAT (Advanced Level) 1128~1131:1128N皇后 1129 模拟推荐系统(set<Node>优化) 1130 中缀表达式
1128 N Queens Puzzle(20 分) 题意:N皇后问题.按列依次给定N个皇后的行号,问N个皇后是否能同时不存在行冲突.列冲突和主副对角线冲突. 分析: 1.根据题意一定不存在列冲突,所 ...
- Redis原理详解
Redis原理详解 数据类型 Redis最为常用的数据类型主要有以下五种: String Hash List Set Sorted set 在具体描述这几种数据类型之前,我们先通过一张图了解下Redi ...
- XV6源代码阅读-中断与系统调用
Exercise1 源代码阅读 1.启动部分: bootasm.S bootmain.c 和xv6初始化模块:main.c bootasm.S 由16位和32位汇编混合编写成的XV6引导加载器.boo ...
- ACM-吴奶奶买鱼
题目描述:吴奶奶买鱼 吴奶奶有个可爱的外孙女——琪琪,她很喜欢小动物,尤其喜欢养鱼.为了让小孙女养到漂亮的小鱼,吴奶奶一大早就到花鸟鱼虫市场买鱼.这个市场可真大,里面有各种各样的宠物,就连宠物鱼都 ...
- UVA - 524 Prime Ring Problem(素数环)(回溯法)
题意:输入n,把1~n组成个环,相邻两个数之和为素数. 分析:回溯法. #pragma comment(linker, "/STACK:102400000, 102400000") ...
- Linux系统发现新恶意软件
导读 安全研究人员发现了一种新的Linux恶意软件,它似乎是由中国黑客创建的,并被用作远程控制受感染系统的手段. 这个恶意软件命名为HiddenWasp,由用户模式rootkit,木马和初始部署脚本组 ...