面试:讲讲 Android 的事件分发机制

写在前面

转眼间 面试系列 已经到了第九期了,由于文章将会持续更新,导致标题难看性,所以以后的标题将更正为本文类似的格式。

好了,话不多说,还是直入主题吧。

面试场景

讲讲 Android 的事件分发机制?

基本会遵从 Activity => ViewGroup => View 的顺序进行事件分发,然后通过调用 onTouchEvent() 方法进行事件的处理。我们在项目中一般会对 MotionEvent.ACTION_DOWNMotionEvent.ACTION_UPMotionEvent.ACTION_MOVEMotionEvent.ACTION_CANCEL 分情况进行操作。

有去查看源码中的事件拦截方法吗?或者说在进行事件分发的时候如何让正常的分发方式进行拦截?

我知道有个拦截事件的方法叫...叫,onInterceptEvent()?应该是,不过由于平时项目较多,确实没时间去关注太多源码。

厄,那你觉得在一个列表中,同时对父 View 和子 View 设置点击方法,优先响应哪个?为什么会这样?

肯定是优先响应子 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() 到底做了什么操作。

直接调用了 DecorViewsuperDispatchTrackballEvent() 方法。DecorView 继承于 FrameLayout,作为顶层 View,是所有界面的父类。而 FrameLayout 作为 ViewGroup 的子类,所以直接调用了 ViewGroupdispatchTouchEvent()

ViewGroup 的事件分发机制

我们通过查看 ViewGroupdispatchTouchEvent() 可以发现。

注意其中红框里面的代码,看注释也能知道,定义了一个 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 的事件分发示意图

  • 事件分发工作流程总结

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在我们的项目中,我们使用了微服务架构来拆分和组织我们的系统。Microservice架构是一种将应用程序拆分为一组小型、独立的服务的方法,每个服务都可以独立部署、扩展和管理。以下是我们在拆分微服务时采取的一些步骤和原则: 1. 领域驱动设计(DDD):我们首先使用领域驱动设计方法来划分业务领域。每个领域都被看作是一个微服务的潜在候选对象。 2. 服务边界划分:我们通过识别业务功能和领域边界来划分服务。我们将相关的业务功能组合在一起,以确保每个服务都有明确的责任和边界。 3. 单一职责原则:我们遵循单一职责原则,即每个微服务应该只负责一个独立的业务功能。这有助于保持服务的内聚性和可维护性。 4. 松耦合和高内聚:我们将注意力放在确保微服务之间的松耦合和高内聚上。松耦合意味着每个服务都应该能够独立开发、测试、部署和扩展,而不会对其他服务产生过多的依赖。高内聚意味着每个服务应该包含与其业务功能相关的所有代码和数据。 5. 服务通信:我们使用轻量级的通信机制,如RESTful API或消息队列,来实现微服务之间的通信。这种方式可以确保松耦合和可扩展性。 6. 持续集成和部署:我们采用持续集成和自动化部署的方法,以便快速、频繁地发布和更新微服务。这有助于加快开发迭代周期,并降低发布新功能的风险。 总的来说,我们在拆分微服务时遵循了领域驱动设计和一些基本的架构原则,以确保每个微服务都具有清晰的责任和边界,并且能够独立开发、测试和部署。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值