肯定是优先响应子 View 的,至于为什么这样,平时知道这个结论,所以没去太深入研究,但我相信我简单看一下源码是肯定知道的。
先发表点扯淡
我们可能经常会遇到上面的这种情况,面试官希望了解我们知识的深入情况,或者说是平时学习欲望到底怎样。可很不幸的是,我搞 模拟面试 以来,80% 的小伙伴都属于开发能力不错,可对类似事件分发这样的基础问题一概不知。究其原因,除去忙以外,大多数小伙伴还是觉得平时开发也用不上什么,即使用到了,直接 Google 一下便能得到正确答案。
这大概就是很多人不会自定义 View 的原因吧,大多数效果在 GitHub 上都是现成的了,即使不太一样,也可以简单改改完事。
可很遗憾的是,我模拟面试那额外的 20% 的人,总拿到了令大多数人羡慕嫉妒恨的 offer,这不是没有原因的。可能别人就平时的开发中保持了更多的一点求知欲,就学到了很多至关重要的细节知识。
正文
还是不能偏题,其实这样的一个面试问题,确实是一个较为普遍的问题,我相信同类型的文章,网上一搜也是比比皆是,而且简单看一下关注度就能知道有多少人倒在了这种源码类型的面试上。
一般情况下,事件列都是从用户按下(ACTION_DOWN)的那一刻产生的,不得不提到,三个非常重要的与事件相关的方法。
-
dispatchTouchEvent()
-
onTouchEvent()
-
onInterceptTouchEvent()
Activity 的事件分发机制
从英文单词中已经很明显的知道,dispatchTouchEvent()
是负责事件分发的。当点击事件产生后,事件首先会传递给当前的 Activity,这会调用 Activity 的 dispatchTouchEvent()
方法,我们来看看源码中是怎么处理的。
注意截图中,我增加了一些注释,便于我们更加方便的理解,由于我们一般产生点击事件都是 MotionEvent.ACTION_DOWN
,所以一般都会调用到 onUserInteraction()
这个方法。我们不妨来看看都做了什么。
很遗憾,这个方法实现是空的,不过我们可以从注释和其他途径可以了解到,该方法主要的作用是实现屏保功能,并且当此 Activity 在栈顶的时候,触屏点击 Home、Back、Recent 键等都会触发这个方法。
再来看看第二个 if 语句,getWindow().superDispatchTouchEvent()
,getWindow()
明显是获取 Window
,由于 Window
是一个抽象类,所以我们能拿到其子类 PhoneWindow
,我们直接看看 PhoneWindows.superDispatchTouchEvent()
到底做了什么操作。
直接调用了 DecorView
的 superDispatchTrackballEvent()
方法。DecorView
继承于 FrameLayout
,作为顶层 View,是所有界面的父类。而 FrameLayout
作为 ViewGroup
的子类,所以直接调用了 ViewGroup
的 dispatchTouchEvent()
。
ViewGroup 的事件分发机制
我们通过查看 ViewGroup
的 dispatchTouchEvent()
可以发现。
注意其中红框里面的代码,看注释也能知道,定义了一个 boolean 值变量 intercept
来表示是否要拦截事件。
其中采用到了 onInterceptTouchEvent(ev)
对 intercept
进行赋值。大多数情况下,onInterceptTouchEvent()
返回值为 false,但我们完全可以通过重写 onInterceptTouchEvent(ev)
来改变它的返回值,不妨继续往下看,我们后面对这个 intercept
做了什么处理。
暂时忽略 判断的 canceled
,该值同样大多数时候都返回 false,所以当我们没有重写 onInterceptTouchEvent()
并使它的返回值为 true 时,一般情况下都是可以进入到该方法的。
继续阅读源码可以发现,里面做了一个 For 循环,通过倒序遍历 ViewGroup
下面的所有子 View,然后一个一个判断点击位置是否是该子 View 的布局区域,当然还有一些其他的,由于篇幅原因,这里就不细讲了。
View 的事件分发机制
ViewGroup
说到底还是一个 View,所以我们不得不继续看看 View 的 dispatchTouchEvent()
。
截图中的代码是有删减的,我们重点看看没有删减的代码。
红框中的三个条件,第一个我就不用说了。
- (mViewFlags & ENABLED_MASK) == ENABLED
该条件是判断当前点击的控件是否为 enable,但由于基本 View 都是 enable 的,所以这个条件基本都返回 true。
- mOnTouchListener.onTouch(this, event)
即我们调用 setOnTouchListener()
时必须覆盖的方法 onTouch()
的返回值。
从上述的分析,终于知道「onTouch()
方法优先级高于 onTouchEvent(event)
方法」是怎么来的了吧。
再来看看 onTouchEvent()
从上面的代码可以明显地看到,只要 View 的 CLICKABLE 和 LONG_CLICKABLE 有一个为 true,那么 onTouchEvent()
就会返回 true 消耗这个事件。CLICKABLE 和 LONG_CLICKABLE 代表 View 可以被点击和长按点击,我们通常都会采用 setOnClickListener()
和 setOnLongClickListener()
做设置。接着在 ACTION_UP 事件中会调用 performClick()
方法,我们看看都做了什么。
从截图中可以看到,如果 mOnClickListener
不为空,那么它的 onClick()
方法就会调用。
总结
本来写到这就结束了,但回顾一遍还是打算给大家稍微总结一下。
需要总结的小点:
1、Android 事件分发总是遵循 Activity => ViewGroup => View 的传递顺序;
2、
onTouch()
执行总优先于onClick()
原本想用文字总结的,结果发现简书上还有这样一篇神文:Android事件分发机制详解:史上最全面、最易懂,所以直接引用一下其中的图片。
- Activity 的事件分发示意图
- ViewGroup 事件分发示意图
- View 的事件分发示意图
- 事件分发工作流程总结
最后分享一波面试复习路线
多余的话就不讲了,接下来将分享面试的一个复习路线,如果你也在准备面试但是不知道怎么高效复习,可以参考一下我的复习路线,有任何问题也欢迎一起互相交流,加油吧!
这里给大家提供一个方向,进行体系化的学习:
1、看视频进行系统学习
前几年的Crud经历,让我明白自己真的算是菜鸡中的战斗机,也正因为Crud,导致自己技术比较零散,也不够深入不够系统,所以重新进行学习是很有必要的。我差的是系统知识,差的结构框架和思路,所以通过视频来学习,效果更好,也更全面。关于视频学习,个人可以推荐去B站进行学习,B站上有很多学习视频,唯一的缺点就是免费的容易过时。
另外,我自己也珍藏了好几套视频,有需要的我也可以分享给你。
2、进行系统梳理知识,提升储备
客户端开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
Android高级架构师
由于篇幅问题,我呢也将自己当前所在技术领域的各项知识点、工具、框架等汇总成一份技术路线图,还有一些架构进阶视频、全套学习PDF文件、面试文档、源码笔记。
- 330页PDF Android学习核心笔记(内含上面8大板块)
-
Android学习的系统对应视频
-
Android进阶的系统对应学习资料
- Android BAT部分大厂面试题(有解析)
好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
[外链图片转存中…(img-rv2lancN-1713772344257)]
- Android BAT部分大厂面试题(有解析)
[外链图片转存中…(img-5YrKhCbU-1713772344258)]
好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!