Android View事件分发机制(一)

在Android中,View的事件分发机制是非常重要的知识体系,掌握View的事件分发机制对我们以后的自定义View和处理View的滑动冲突等问题有很大的帮助;
我们这里主要分析的对象就是MotionEvent,即点击事件,所谓的点击事件分发就是对MotionEvent事件的分发过程。点击事件的分发过程由三个非常重要的方法共同来完成:
1、dispatchTouchEvent:用来进行事件的分发,如果事件能够传递给当前View,那么此方法一定会调用;
2、onInterceptTouchEvent:用来判断是否拦截某个事件;如果当前View拦截了某个事件,那么在同一个事件序列当中,此方法不会再次调用;
3、onTouchEvent:用来处理响应点击事件;
注:子View中是没有onInterceptTouchEvent方法的,因为View已经是最底层了,无须再向下传递了,只有ViewGroup当中有onInterceptTouchEvent方法;
接下来我们就用一个实例来说明一下事件的传递过程,首先我们创建一个布局由Activity、ViewGroup(LinearLayout)、View(TextView)三层控件组成,然后我们点击View(TextView)来看一下View的事件传递流程:
log日志输出如下:

 Activity____dispatchTouchEvent
 ViewGroup___dispatchTouchEvent
 ViewGroup___onInterceptTouchEvent
 View________dispatchTouchEvent
 View________onTouchEvent
 ViewGroup___onTouchEvent
 Activity____onTouchEvent

通过我们测试的输出日志我们可以看到(测试中我们默认对事件均不处理),在收到一个点击事件后,最外层的Activity首先收到,执行Activity的dispatchTouchEvent开始事件传递,然后传递给ViewGroup的dispatchTouchEvent,ViewGroup的dispatchTouchEvent收到后,调用ViewGroup的onInterceptTouchEvent,询问是否要进行事件拦截,然后不拦截继续向下传递,传递给View的dispatchTouchEvent,View的dispatchTouchEvent收到事件后默认调用View的onTouchEvent方法,即开始事件的处理,默认不处理,然后向上传递给ViewGroup的onTouchEvent,ViewGroup的onTouchEvent也是默认不处理,最后传递给Activity的onTouchEvent来进行处理;
接下来我们来看一下上例当中的事件传递图


点击事件传递流程图

上图清晰的展示出了在我们默认不做任何处理的情况下,点击事件传递的整个事件链,最外层的控件先收到点击事件然后一层一层向下传递,传递到最底层后开始事件的处理,最底层的控件先开始事件处理,然后一层一层向上传递;因此我们可以得到结论:
1、事件的传递 由外向内
2、事件的处理 由内向外
接下来我们来举更多的例子,把各种情况都考虑进去,来开一下点击事件是如何传递的,以便让大家彻底了解View的事件传递机制;

情况1:View的dispatchTouchEvent返回true(拦截事件传递)

log输出:

 Activity____dispatchTouchEvent
 ViewGroup___dispatchTouchEvent
 ViewGroup___onInterceptTouchEvent
 View________dispatchTouchEvent

图示:

情况2

可以看到事件从外向内传递,dispatchTouchEvent负责传递事件,当传递到最里层View的dispatchTouchEvent,如果我们返回true,则表示结束传递消费该事件;所有View的onTouchEvent均不再执行;
结论:dispatchTouchEvent返回true将会直接消费该事件,并且中断事件传递,onTouchEvent方法也不会被调用

情况2:ViewGroup的onInterceptTouchEvent返回true拦截传递

log输出:

 Activity____dispatchTouchEvent
 ViewGroup___dispatchTouchEvent
 ViewGroup___onInterceptTouchEvent
 ViewGroup___onTouchEvent
 Activity____onTouchEvent

图示:

情况2

可以看到ViewGroup的onInterceptTouchEvent返回true表示拦截事件传递,事件将不会再向下传递,并且会调用ViewGroup的onTouchEvent来响应处理事件,如果ViewGroup的onTouchEvent默认不处理事件,将会继续向上调用Activity的onTouchEvent来处理事件;
结论:onInterceptTouch返回true表示拦截事件传递,并调用自身的onTouchEvent方法,下层控件将不会再收到相应事件;

情况3:View的dispatchTouchEvent返回false

log输出:

 Activity____dispatchTouchEvent
 ViewGroup___dispatchTouchEvent
 ViewGroup___onInterceptTouchEvent
 View________dispatchTouchEvent
 ViewGroup___onTouchEvent
 Activity____onTouchEvent

图示:


情况3

可以看到View的dispatchTouchEvent返回false表示我不响应该事件,并回调父控件的onTouchEvent方法;
结论:dispatchTouchEvent返回false表示不拦截并且不响应该事件,并回调父控件的onTouchEvent方法让父控件去处理事件;

情况4:View的onTouchEvent返回true

log输出:

 Activity____dispatchTouchEvent
 ViewGroup___dispatchTouchEvent
 ViewGroup___onInterceptTouchEvent
 View________dispatchTouchEvent
 View________onTouchEvent

图示:

情况4

可以看到当View的onTouchEvent方法返回true时就表示要消费该事件,并停止传递;
结论:onTouchEvent返回true表示消费该事件,同时终止事件传递;
综上所有情况总结:
1、dispatchTouchEvent 返回 true 表示消费该事件,终止事件传递,所有的onTouchEvent方法都不会执行;
2、dispatchTouchEvent 调用 super 向下传递事件,调用子View的dispatchTouchEvent,若没有子View了,调自身的onTouchEvent方法;
3、dispatchTouchEvent 返回 false 表示终止事件传递,不再向下传递事件,并且不会调用自身的onTouchEvent方法,而是向上调用父控件的onTouchEvent方法;
4、onInterceptTouchEvent 返回 true 表示拦截事件,不会再向下传递事件,然后调用自身的onTouchEvent方法;
5、onInterceptTouchEvent 返回 false/super 表示不拦截事件,继续向下传递事件;
6、onTouchEvent 返回 true 表示消费该事件,事件传递终止;
7、onTouchEvent 返回 false/super 不处理,事件继续向上传递;
有了以上总结之后,Android中的View点击事件传递流程已经非常清晰了,因此我画了一个整体的事件分发流程图,如下:
事件分发流程图

以上就是View事件分发的整体流程图,相信经过上面的总结之后你会对View的事件分发有更深的认识;但是以上只是针对触摸事件的Down(单击)事件来分析的,但是实际情况中,我们还要对触摸事件的Move和Up事件进行处理,所以在下一篇文章我们将深入分析在Move和Up的情况下事件的分发机制;

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352