(一文弄懂!!超详细!!)安卓事件分发机制-程序员宅基地

技术标签: 安卓  移动开发(AndroidStudio)  android  软件工程  

一、安卓事件分发的整个流程

Android 的事件分发流程是一个从顶至底的层级传递过程,涉及 Activity、ViewGroup 和 View。这个流程确保了触摸事件(如屏幕触摸操作)能够被正确地处理和响应。我们将首先概括一下一次事件触发的整个流程,再逐一对其中的细节进行分别介绍:

1. 事件生成:

  • 当用户触摸屏幕时,底层的硬件和操作系统生成一个触摸事件(如 MotionEvent)。

2. InputManager 处理:

  • InputManager 接收这个事件,并确定哪个窗口(通常对应一个 Activity)应该接收这个事件。

3. Activity 分发:

  • 事件首先到达 Activity,通过 Activity 的 dispatchTouchEvent() 方法开始处理。
  • 如果 Activity 没有完全拦截这个事件,它将事件传递给它的根视图(一个 ViewGroup)。

4. ViewGroup 分发:

  • 在 ViewGroup 中,dispatchTouchEvent() 同样被调用。
  • ViewGroup 可以选择拦截这个事件,通过实现 onInterceptTouchEvent() 方法。如果 ViewGroup 拦截了事件(返回 true),则事件不再向下传递,而是交由 ViewGroup 的 onTouchEvent() 方法处理。
  • 如果 ViewGroup 没有拦截(onInterceptTouchEvent() 返回 false),事件将继续传递给它的子视图。

5. 子视图处理:

  • 事件接着被传递到子视图(可能是另一个 ViewGroup 或一个 View)的 dispatchTouchEvent()
  • 这一过程递归进行,直到找到最终的事件接收者。

6. 事件响应:

  • 最终的视图(View)通过它的 onTouchEvent() 方法处理这个事件。
  • 如果这个视图没有处理(onTouchEvent() 返回 false),事件会回溯到它的父 ViewGroup,以此类推,直到找到能处理这个事件的视图。

7. 事件消费:

  • 一旦事件被一个视图的 onTouchEvent() 方法处理并消费(方法返回 true),事件分发流程结束。这个视图成为了该触摸序列(如一次点击或滑动操作)的消费者,后续的事件(如触摸移动或抬起)也会直接分发给这个视图。

这个流程确保了 Android 应用可以灵活处理用户的触摸输入,同时也提供了多个层级对事件进行截获和响应的能力。
接下来我们将分别对InputManager事件对象涉及的方法以及事件类型逐一进行详细讲解。

二、InputManager

InputManager 在 Android 系统中是一个核心的服务组件,负责管理和分发从各种输入设备(如触摸屏、键盘、鼠标等)来的输入事件。Android 的输入系统架构中处于较为底层的位置,它不仅负责将输入事件从硬件传递到应用,还处理了输入事件的多种复杂场景它在,主要功能和特点包括:

1. 事件接收和处理:

  • InputManager 负责从底层的 Linux 内核接收原始的输入事件。这些事件可能来自触摸屏、物理键盘、鼠标、游戏手柄、陀螺仪等多种输入源。
  • 它对这些原始事件进行预处理,如合成、转换或调整事件数据。

2. 事件分发:

  • 经过处理后,InputManager 将这些事件分发到正确的窗口和视图(View)。这个分发过程涉及判断哪个应用或窗口应该接收事件,并将事件传递给它。
  • 在这个过程中,InputManager 会考虑焦点控制、窗口的布局和层级关系等因素。

3. 多点触控和手势识别:

  • InputManager 能够处理复杂的多点触控数据,识别并分发与多点触控相关的手势事件。
  • 它为应用提供了一个相对简化的接口来处理这些高度复杂的输入。

4. 设备管理:

  • InputManager 还负责管理输入设备,提供查询连接的输入设备、它们的能力和状态等功能。

5. 与应用层的交互:

  • 虽然 InputManager 主要在系统层面运作,但它为应用层提供了接口,使得应用可以查询输入设备的信息、注册以接收特定设备的事件等。
  • 例如,游戏应用可能需要查询游戏手柄的状态或注册接收其输入事件。

6. 安全性和权限管理:

  • InputManager 在处理输入事件时,也需要考虑安全性和权限。例如,它需要确保敏感的输入数据(如密码输入)不会被未授权的应用捕获。
    总的来说,InputManager 是 Android 输入系统的核心,它不仅负责将输入事件从硬件传递到应用,还处理了输入事件的多种复杂场景,确保了输入事件的高效和安全处理。

三、三个事件传递对象

在Android中,事件分发机制主要涉及三个层级的对象:Activity、ViewGroup和View。它们的主要区别如下:

1. Activity:

  • Activity是Android应用中的一个重要组件,代表了一个屏幕上的一个单独的窗口。
  • 它不直接参与触摸事件的处理,但是是事件分发的起点。当一个触摸事件发生时,系统首先将事件传递给当前处于活动状态的Activity。
  • Activity通常包含一个或多个ViewGroup和View,负责整个应用界面的布局和显示。

2. ViewGroup:

  • ViewGroup是一个抽象类,继承自View,是所有布局容器的基类,如LinearLayout、RelativeLayout等。
  • 它不仅是View的容器,而且还负责管理其子View(包括其他ViewGroup)的布局和绘制。
  • 在事件分发过程中,ViewGroup有能力决定是否拦截某个事件(通过onInterceptTouchEvent()方法),如果不拦截,事件则会继续向下传递给其子View。

3. View:

  • View是Android中所有用户界面组件的基类,如按钮、文本框等。
  • View是用户交互的基本单元,用于绘制和处理用户的交互事件,如触摸、点击。
  • 在事件分发机制中,View是事件链的最末端,如果事件传递到了View,它将通过onTouchEvent()方法来处理这个事件。

四、涉及的方法

在Android的事件分发过程中,主要涉及以下几个方法:

1. dispatchTouchEvent():

  • 这是事件分发的起始方法。
  • 在Activity、ViewGroup和View中都有这个方法。
  • 它负责将触摸事件(MotionEvent)分发到正确的子View或者自身。
  • 在ViewGroup中,它还会决定是否调用自己的onInterceptTouchEvent()方法来判断是否要拦截事件。

2. onInterceptTouchEvent():

  • 这个方法只存在于ViewGroup中。
  • 它用于决定当前的ViewGroup是否应该拦截触摸事件。
  • 如果这个方法返回true,表示ViewGroup决定拦截并处理这个事件,事件将不会继续向下传递。
  • 如果返回false,事件会继续传递给子View。

3. onTouchEvent():

  • 这个方法存在于Activity、ViewGroup和View中。
  • 它用于处理触摸事件。
  • 如果事件未被拦截且当前View是事件传递链上的最后一层,该事件最终会传递到这里。

4. dispatchKeyEvent():

  • 虽然主要用于键盘事件的分发,但也是事件分发机制的一部分。
  • 它类似于dispatchTouchEvent(),用于分发按键事件。

5. onKeyDown() 和 onKeyUp():

  • 这些方法用于处理按键事件的按下和释放。
  • 它们存在于Activity和View中。

6. onKeyLongPress():

  • 这个方法用于处理长按键事件。
  • 它通常存在于Activity和View中。

五、涉及的事件类型

1. 触摸事件(Touch Events)

  • MotionEvent 是处理触摸事件的主要类。它包含了诸如触摸屏幕的动作类型(如按下、移动、抬起等)、触摸点的位置、压力、时间戳等信息。
  • 触摸事件的动作类型主要包括 ACTION_DOWN(按下)、ACTION_MOVE(移动)、ACTION_UP(抬起)、ACTION_CANCEL(取消)等。

2. 按键事件(Key Events)

  • KeyEvent 用于处理按键事件,比如用户按下或释放手机上的物理按键(如音量键、返回键等)。
  • 主要的动作类型包括 ACTION_DOWN(按键按下)和 ACTION_UP(按键释放),以及 ACTION_MULTIPLE(多个连续的按键动作)。

3. 滚动事件(Scroll Events)

  • 涉及到滚动操作,如列表滚动。这些通常是由触摸事件触发的,但它们有自己的特殊处理逻辑。

4. 长按事件(Long Press Events)

  • 当用户长时间按住屏幕的某个点时触发。虽然这是基于触摸事件的,但通常需要特殊的处理来区分普通的触摸事件。

5. 焦点变化事件(Focus Change Events)

  • 当用户界面中的焦点发生变化时触发,例如从一个文本框移动到另一个文本框。

6. 手势事件(Gesture Events)

  • 如双指缩放、旋转等。这些事件通常是由多个触摸点组合起来的复杂动作。

7. 其他特定于设备的事件

  • 如陀螺仪事件、加速度计事件等,这些通常用于高级的用户界面交互,例如在游戏或增强现实应用中。
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/Moonsev/article/details/135919109

智能推荐

2019校招面经大汇总-程序员宅基地

文章浏览阅读191次。转载链接:https://www.nowcoder.com/discuss/90907?type=0&order=3&pos=24&page=1【杭州有赞】 【Java】杭州有赞三面技术面试https://www.nowcoder.com/discuss/85395 【多益网络】 【产品研发工程师】多益网络 提前批..._面经vivo2018年校招-简书

git commit提交报错subject may not be empty [subject-empty]-程序员宅基地

文章浏览阅读2.3w次,点赞10次,收藏7次。今天在sourcetree提交写好的代码突然报错,显示如下错误: subject may not be empty [subject-empty] type may not be empty [type-empty]_subject may not be empty

FS102WS是一款用于 LED 灯光开关控制及亮度调节的触摸IC-程序员宅基地

文章浏览阅读187次。每次短按触摸,依OPT1/2/3选择不同,灯光亮度按[高亮度→>中亮度->低亮度->灭]依次循环变化,或[低亮度->中亮度->高亮度->灭]依次循环变化。FS102WS是一款用于 LED 灯光开关控制及亮度调节的触摸IC,支持单通道触摸输入、单路 PWM 输出,可在有介质(如玻璃、亚克力、塑料、陶瓷等)隔离保护的情况下实现触摸功能,可靠性非常高,灯光无闪频。未断电短按开灯后第一次长按调光的方向由之前记忆的亮度值来决定,若记忆亮度值大于45%,则向下调光:若记忆亮度值小于45%,则向上调光。

Python办公自动化实战 09 | Python-docx库:Python与Word的完美结合_ 如何在Word中生成表格?把Python办公自动化进行到底-程序员宅基地

文章浏览阅读911次。本小节主要演示了怎么向Word文档中创建表格并插入数据,并且对表格格式做个性化的设定。_python与word的完美结合

MySQL:从库binlog 使用mysqlbinlog stop-datetime过滤问题-程序员宅基地

文章浏览阅读2k次。更多主从同步相关可以参考我的《深入理解MySQL主从原理》专栏:本文是一个朋友问我问题。从库使用mysqlbinlog..._mysql stop-datetime

SAP入门经验_sap经验-程序员宅基地

文章浏览阅读8k次,点赞18次,收藏30次。SAP入门的经验SAP业务顾问入门确实起点比较高,这在我最开始入门的时候不以为然,但是随着学习的深入,才发现原来老师们说的是真的!简单说一下我自己的入门经历,我是本科是工业工程(IE)专业的,如果有了解的肯定知道这个专业是干什么的,步入这个行业我才发现我所学的专业知识都挺有用的,特别是PP模块,我本身在大学就经常参加一些生产优化案例竞赛,对于排产,MRP等信息有了初步的了解,更重要的是IE专业培养了我的优化意识我感觉这是我的一大笔财富。好了步入正题,说一下我的入门经历:最开始公司培训讲了很多模块的知_sap经验

随便推点

探索GeoTIFF.js:地图数据处理的新利器-程序员宅基地

文章浏览阅读422次,点赞5次,收藏7次。探索GeoTIFF.js:地图数据处理的新利器项目地址:https://gitcode.com/geotiffjs/geotiff.jsGeoTIFF是一种用于存储地理空间信息的TIFF文件格式,它结合了图像数据和元数据,为遥感和GIS领域的数据分析提供了强大支持。GeoTIFF.js是一个JavaScript库,旨在简化Web应用中对GeoTIFF文件的操作。这篇文章将深入探讨GeoTIFF..._js地图数据

jetson nano ubuntu 安装配置opencv4 cuda10 pytorch_nano 安装cuda10-程序员宅基地

文章浏览阅读1.1k次,点赞2次,收藏14次。安装cudabashrc文件添加如下export PATH=/usr/local/cuda/bin:$PATHexport LD_LIBRARY_PATH=/usr/local/cuda/lib64:$LD_LIBRARY_PATH查安装nvcc -V卸载自带opencvsudo apt-get purge libopencv*sudo apt autoremove下载opencv与opencv_contrib版本对应opencv下载opencv_contrib下载编译安装_nano 安装cuda10

【计算机毕设选题】基于机器视觉的二维码识别检测 - opencv 二维码 识别检测 机器视觉_基于opencv的二维码识别课程设计-程序员宅基地

文章浏览阅读733次,点赞19次,收藏23次。今天学长向大家介绍一个机器视觉的毕设项目,二维码 / 条形码检测与识别基于机器学习的二维码识别检测 - opencv 二维码 识别检测 机器视觉。_基于opencv的二维码识别课程设计

互联网医院智慧医院系统_互联网可访问的医院系统有哪些-程序员宅基地

文章浏览阅读8.5k次。什么是互联网医院?我们来看一下百度百科的定义“互联网医院系统”带有在线问诊、随访、慢病管理等功能,它有实体医院作强有力的支撑,线上方便病人,就是简单的问题不需要到医院,在网上就可以进行。可见在线问诊、随访、慢病管理是互联网医院的核心功能,下面我们来逐个分析一下在线问诊:一般是图文问诊、视频问诊、语音问诊、电话问诊几种问诊方式,多种方式方便医患在线沟通随访:医生可以对患者发起线上随访,患者可在线填写随访表单,后台可统计随访结果慢病管理:能对糖尿病、高血压等常见慢性病进行健康管理,医生可以随时查看病人_互联网可访问的医院系统有哪些

IOS开发 生成app图标_ios app图标生成-程序员宅基地

文章浏览阅读1.6k次。IOS开发,给app设置图标_ios app图标生成

[Python] Django 报错记录与解决_importerror: cannot import name 'iterable' from 'c-程序员宅基地

文章浏览阅读6.7k次,点赞3次,收藏17次。配置记录pycharm中打开Django项目并配置虚拟环境运行项目;将Django项目全局配置文件用统一的包进行管理;配置jinja2模板引擎;补充 Jinja2 模板引擎环境报错记录ImportError;You must set settings.ALLOWED_HOSTS if DEBUG is False;'cryptography' package is required;ImproperlyConfigured_importerror: cannot import name 'iterable' from 'collections

推荐文章

热门文章

相关标签