宝贝到底归谁?(Android View事件分发机制)

2023-12-23 21:08

本文主要是介绍宝贝到底归谁?(Android View事件分发机制),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

    在《Android开发艺术探索》一书中,作者大篇幅的讲解了关于View的事件体系。对于这一章节的内容,我有较多想要叙述和温习的地方,所以,在以下的文章中,我会着重与阐述关于整个View和ViewGroup的事件分发机制以及我对他们的理解。

一、什么是事件
首先,了解Android中View的事件分发机制,就必须先知道什么是事件?在手机软件中,每个不同部分的内容或者组件几乎都有一项基础的功能,那就是点击,拖动,长按,等等。这些动作,会被Android系统进行拦截,并且拆分成不同的动作类型,而后交由各个对应的组件完成处理,最终处理结果反馈给用户。那么,我们可以粗略的认为点击,拖动,长按,等等用户产生的与屏幕点击有关的动作就是事件,这一系列的动作就被成为一个完整的事件。事件的产生也就推动了整个Android系统分发机制的产生和完善。

二、事件拦截,事件分发
既然有了一整套完整的事件,那么就会产生几个问题,一个界面上有许许多多的组件,他们都可以对这一整套事件进行拦截和处理,那么应该交给谁呢?谁又是真正需要这整套事件的组件呢?这些问题将由Android系统中的事件分发机制来进行解答。在详细描述事件分发机制前,我要先明确一个概念,那就是事件拦截。因为有了事件拦截,所以手机中的各个组件也才能捕捉得到整个事件。也是通过事件拦截,才能判断这整套事件应该有谁处理,谁想要进行处理。于是,对应着事件拦截,现在我要隆重介绍这个函数,那就是public boolean onInterceptTouchEvent(MotionEven ev)。如果把整个事件理解为一件宝物,那么对于这个函数而言,他就是宝物的竞拍人,他会对这个宝物(事件)进行分析,看看自己对宝物有没有兴趣,看看自己有没有能力将宝物收入囊中,如果有兴趣,有能力,则会亮出自己的竞价牌(返回true),否则他将放弃竞争机会(返回false),而后将宝物收入自己囊中(交由家里人进行处理)。

   现在就会有两种情况发生,第一种情况,如果第一个竞价人(View组件)顺利的拿下了这件宝物(拦截事件,onInterceptTouchEvent 函数返回true),则他会将这一系列的点击事件交给家中人进行进一步的处理。第二种情况,如果第一个竞价人没有意愿或者没有能力拿下这一宝物(不拦截事件,onInterceptTouchEvent 函数返回false),那么拍卖行(系统)就会询问下一个竞拍人(下一层的View组件),以此类推,直到找到需要或者有能力拿下宝物的人出现,如果问遍了拍卖行中的所有竞拍人,都没有人回应,那么,拍卖行就会自行将其进行处理。在拍卖行进行逐个人询问时,如何来确定询问的那个人是否有资格来进行竞拍呢?这个时候,就会需要每个竞拍人都要挂有一个入场证,也就是函数 public boolean dispatchTouchEvent(MotionEvent ev)。只要有这个牌子,那么这个人就可以进场参加拍卖。而且每次在上个竞拍人(上层View组件)没有成交后(未拦截事件),拍卖行都会先查看下个拍卖人的挂牌(调用dispatchTouchEvent 函数),之后,再询问其是否会竞拍该宝物。

三、事件处理
在拍卖人把宝物拿回家之后,就会按照家中辈分顺序来对这件宝物进行处理,首先是大哥onTouch(),如果大哥能把这件宝物处理好,那么他就会告诉大家,宝物在我这里处理OK啦(onTouch返回true)。那么后边的二哥onTouchEvent()便不会再过问。交易行也会得知这一消息,并且不再挂出该宝物(该事件信息)。

   但是,如果大哥发现这件宝物在他这里没有处理好,那么他就会告诉大家,哎鄙人没办法处理这个宝物啊喂(onTouch返回false)。那么,这个宝物(事件)就会交由二哥继续处理,同样的,二哥如果能够处理好那么他也会告诉大家(onTouchEvent返回true)。如果没能处理好(返回false),那么,他会把这件宝物交还个拍卖行。不好意思,另请高明吧。那么,拍卖行又会继续把这件宝物向接下来的竞拍人兜售。直到宝物被接收并且安放妥当。这就是View中的事件分发过程,这只是在View层进行了讲解。接下来我们要了解另一个东西,那就是ViewGroup。通过英语直译我们就能直到,这货肯定是View们的组长了,他是承载所有View的容器。你想,管理了那么多的View,那么事件分发时,能少了他,能怠慢他吗?那显然是不可能的,所以,我们一定要记住了有好东西,一定要先想着老大哥,有了老大哥一口肉吃,才有你们一口汤喝(点击事件首先要经过ViewGroup,通过ViewGroup才能向上层的View传递)- _ -!,人在屋檐下不得不低头,连代码界都是如此。不过老大哥(ViewGroup)还是很无私的,他不会主动截留宝物(事件),而是把宝物通过拍卖行询问小弟们(各个子View组件)是否需要。如果老大哥(ViewGroup)需要这件宝物,那么他留下之后,便不会再给小弟们一点得到这件宝物的机会了。

四、总结
好了,了解了这几个必要的原理之后,那么我们现在就需要对这个原理进行总结,首先,再点击事件被系统捕捉之后,会将其传递给Activity,而后Activity再将这个事件传递给Window,也就是PhoneWindow对象,通过这个对象找到顶级View,也就是ViewGroup,而后再从ViewGroup出发,遍历整个子View布局,从而找到能够消耗这一个事件的View。

   当View接收到事件之后,就会将其进行相应的处理,一般为onTouch()到onTouchEvent()。而我们熟知的onClick()函数,其实就是在onTouchEvent()中被调用的,他是位于整个事件消耗的尾端,当所有动作都不消耗事件时,他才会被用来判断是否需要事件。同时,我们也需要注意的是,只有当前View组件可以点击同时有onClickListener时,我们才能使用onTouch()来进行业务逻辑的书写。否则,我们应使用onTouchEvent()进行代码编写。这就是我对于View事件分发体系宏观上的理解,如果有理解错误或者笔误,请大家指出。如果想要从源码角度理解这一机制,可以参看郭霖大神的博客:http://blog.csdn.net/guolin_blog/article/details/9153747

这篇关于宝贝到底归谁?(Android View事件分发机制)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/529465

相关文章

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st

Android多线程下载见解

通过for循环开启N个线程,这是多线程,但每次循环都new一个线程肯定很耗内存的。那可以改用线程池来。 就以我个人对多线程下载的理解是开启一个线程后: 1.通过HttpUrlConnection对象获取要下载文件的总长度 2.通过RandomAccessFile流对象在本地创建一个跟远程文件长度一样大小的空文件。 3.通过文件总长度/线程个数=得到每个线程大概要下载的量(线程块大小)。

vue+elementui分页输入框回车与页面中@keyup.enter事件冲突解决

解决这个问题的思路只要判断事件源是哪个就好。el分页的回车触发事件是在按下时,抬起并不会再触发。而keyup.enter事件是在抬起时触发。 so,找不到分页的回车事件那就拿keyup.enter事件搞事情。只要判断这个抬起事件的$event中的锚点样式判断不等于分页特有的样式就可以了 @keyup.enter="allKeyup($event)" //页面上的//js中allKeyup(e

Linux系统稳定性的奥秘:探究其背后的机制与哲学

在计算机操作系统的世界里,Linux以其卓越的稳定性和可靠性著称,成为服务器、嵌入式系统乃至个人电脑用户的首选。那么,是什么造就了Linux如此之高的稳定性呢?本文将深入解析Linux系统稳定性的几个关键因素,揭示其背后的技术哲学与实践。 1. 开源协作的力量Linux是一个开源项目,意味着任何人都可以查看、修改和贡献其源代码。这种开放性吸引了全球成千上万的开发者参与到内核的维护与优化中,形成了

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

高仿精仿愤怒的小鸟android版游戏源码

这是一款很完美的高仿精仿愤怒的小鸟android版游戏源码,大家可以研究一下吧、 为了报复偷走鸟蛋的肥猪们,鸟儿以自己的身体为武器,仿佛炮弹一样去攻击肥猪们的堡垒。游戏是十分卡通的2D画面,看着愤怒的红色小鸟,奋不顾身的往绿色的肥猪的堡垒砸去,那种奇妙的感觉还真是令人感到很欢乐。而游戏的配乐同样充满了欢乐的感觉,轻松的节奏,欢快的风格。 源码下载