当前位置: 首页 > news >正文

android 输入系统

一、输入系统的核心角色与分层架构

Android 输入系统的本质是桥梁:一端连接硬件驱动产生的原始事件,另一端将事件精准派发给应用窗口。整个过程涉及三层架构和多个关键组件,可类比为 “快递分拣系统”:

1. 硬件与内核层(源头)
  • 角色:当用户触摸屏幕或按下按键时,硬件驱动将事件写入设备节点(如/dev/input),生成原始的内核事件(类似 “快递包裹的原始数据”)。
  • 技术实现:通过EventHub(事件枢纽)监听设备路径,使用epoll和inotify机制高效检测事件变化和设备插拔。

系统内容:

驱动上报

struct RawEvent{

nsecs_t when;

nsecs_t readtime;

int32_t deviceId; 输入设备唯一标识符

int32_t type;  事件类型 如EV_KEY, EV_ABS

int32_t code;  事件码

int32_t value; 事件值

}

frameworks/native/services/inputflinger/reader/include/EventHub.h

2. Native 层(事件处理与分发)

InputReader(事件快递员) :

    • 从EventHub读取原始事件(如触摸坐标、按键码),按规则封装为标准事件(如MotionEvent、KeyEvent)。
    • 类比:将 “原始包裹数据” 解析为 “标准化快递单”。

InputDispatcher(事件分拣员) :

    • 接收InputReader处理后的事件,结合窗口信息(如焦点窗口),将事件派发给对应的应用窗口。
    • 类比:根据 “快递单地址” 将包裹分拣到正确的配送路线。

InputManager(调度中心) :

    • 管理InputReader和InputDispatcher,创建并启动它们的工作线程。

  1. 按键事件类型

RawEvent的 type ==EV_KEY:

rawEvent 的code 对应android的scanCode,

scanCode  通过  Generic.kl 映射到android 的keycode。

frameworks/base/data/keyboards/Generic.kl

  1. 触摸事件类型

RawEvent的type ==EV_ABS (绝对坐标事件)

在 Linux 输入子系统(Input Subsystem)中,多点触控(Multi-Touch)事件通过一系列以 ABS_MT_ 开头的绝对坐标类事件代码(ABS 代表 Absolute Position)来描述每个触摸点(Slot)的属性。以下是你列出的各个 ABS_MT_ 事件的详细说明:

1. ABS_MT_SLOT

  • 作用:标识当前操作的触摸点槽位(Slot)。
    多点触控设备通过 “槽位” 机制管理多个触摸点(类似数组索引),每个槽位对应一个独立的触摸点。当设备报告某个触摸点的属性时,需先通过 ABS_MT_SLOT 指明操作的是哪个槽位。
  • 值范围:通常从 0 开始递增(如 0、1、2...),具体取决于设备支持的最大触摸点数(如 5 点触控则槽位为 0~4)。

2. ABS_MT_TRACKING_ID

  • 作用:为每个触摸点分配唯一的追踪 ID,用于在触摸点生命周期内(按下、移动、抬起)标识其身份。
    • 当触摸点按下时,系统分配一个非负整数 ID(如 1、2...);
    • 当触摸点抬起时,ID 会被重置为 -1(表示该槽位不再被占用)。
  • 值范围:
    • 有效触摸点:>= 0(如 1, 2);
    • 无效 / 释放的槽位:-1。
  • 用途:区分不同触摸点(即使槽位重用),例如:
    • 槽位 0 先用于触摸点 A(ID=1),抬起后 ID 重置为 -1;
    • 新触摸点 B 按下时,可能再次使用槽位 0,但分配新 ID=2。
      通过 ID 可确保触摸点的移动轨迹不会因槽位重用而混淆。

3. ABS_MT_TOUCH_MAJOR

  • 作用:表示触摸点接触面积的主轴长度(椭圆的长轴,单位为像素或设备特定单位)。
    可粗略理解为触摸点的 “宽度” 或 “接触区域大小”,例如手指按下时的接触面积。
  • 值范围:通常为正整数,值越大表示接触面积越大。
  • 示例:手指轻轻触摸时值为 20,用力按下时值为 30

4. ABS_MT_WIDTH_MAJOR

  • 作用:表示触摸点接触面积的次轴长度(椭圆的长轴,单位与 ABS_MT_TOUCH_MAJOR 一致)。
    在某些设备中,TOUCH_MAJOR 和 WIDTH_MAJOR 可能分别对应椭圆的长轴和短轴,用于描述触摸点的形状。
  • 值范围:正整数,通常与 ABS_MT_TOUCH_MAJOR 成比例。

5. ABS_MT_POSITION_X 和 ABS_MT_POSITION_Y

  • 作用:
    • ABS_MT_POSITION_X:触摸点在屏幕坐标系中的 X 轴坐标(水平位置)。
    • ABS_MT_POSITION_Y:触摸点在屏幕坐标系中的 Y 轴坐标(垂直位置)。
  • 坐标原点:通常为屏幕左上角(X=0, Y=0),向右 / 向下递增。
  • 单位:设备特定的逻辑单位(如像素、毫米等),需通过输入子系统校准后映射到屏幕像素。

6. ABS_MT_PRESSURE

  • 作用:表示触摸点的压力值,用于检测触摸力度(如手指按下的轻重)。
    • 值为 0 时表示无压力(触摸点抬起);
    • 值越大表示压力越大。
  • 值范围:通常为 0 到设备支持的最大值(如 255、1024 等)。
  • 用途:实现压感功能,例如绘图应用中根据压力调整笔触粗细。

3.EV_SYN(同步事件)​​

  • ​​核心作用​​:标记事件数据包的边界,确保用户空间程序能完整处理一组事件

·  ​​子类型​​:

  • ​​SYN_REPORT​​:表示当前数据包结束,触发用户空间处理累积事件(如鼠标移动后必须发送该事件完成坐标更新)

·  ​​SYN_DROPPED​​:内核缓冲区溢出时通知用户丢弃数据包并重新查询设备状态

·  ​​SYN_MT_REPORT​​:多点触控协议中分隔不同触点的数据包(Type A协议使用)

·  ​​底层依赖​​:驱动必须正确发送该事件,否则用户空间无法识别事件边界

4.EV_REL(相对坐标事件)​​

  • ​​功能​​:报告相对位移变化,适用于鼠标等设备

5.EV_SW(开关事件)​​

  • ​​功能​​:报告二进制状态切换,如设备休眠/唤醒、盖子开合等

6.EV_MSC(杂项事件)​​

  • ​​功能​​:处理无法归类到其他类型的事件,如硬件特定状态或补充信息

3. Java 层(系统服务与交互)

InputManagerService(IMS,总控中心) :

    • 作为 Android 系统服务(运行于system_server进程),通过 JNI 与 Native 层交互。
    • 与窗口管理服务(WMS)同步窗口信息,为InputDispatcher提供派发依据(如哪个窗口当前可见)。
    • 类比:“快递总控中心”,协调底层分拣与上层应用的对接。

二、启动流程详解:从 IMS 初始化到线程启动

IMS 的启动伴随system_server进程启动,整个过程可分为对象创建线程启动两个阶段,涉及 Java 层、JNI 层和 Native 层的跨层调用。

1. 初始化阶段:搭建组件链路

/ /Java层:IMS初始化(InputManagerService.java)inputManager = new InputManagerService(context);

步骤 1:创建 Java 层 IMS 对象

初始化mHandler,运行在 “android.display” 线程(负责处理 Java 层消息)。

通过nativeInit调用 JNI,进入 Native 层初始化。

// JNI层:nativeInit(com_android_server_input_InputManagerService.cpp)NativeInputManager* im = new NativeInputManager(...);

步骤 2:创建 NativeInputManager(JNI 桥梁)

持有 Java 层 IMS 对象的引用(mServiceObj),作为 Native 层与 Java 层交互的桥梁。

创建EventHub(监听设备事件)和InputManager(管理读写线程)。

// Native层:InputManager构造(InputManager.cpp)

mDispatcher = new InputDispatcher(...); // 分拣员

mReader = new InputReader(...); // 快递员

步骤 3:创建 InputDispatcher 和 InputReader

  • InputDispatcher关联NativeInputManager(获取派发策略,如超时参数)。
  • InputReader通过QueuedInputListener与InputDispatcher建立连接(事件传递的枢纽)。

2. 启动阶段:激活工作线程

inputManager.start(); // 调用nativeStart

· 启动 Native 层线程
通过InputManager.start()启动两个核心线程:

  • ·  InputReaderThread:循环调用EventHub.getEvents()读取事件,交由InputReader处理。
  • InputDispatcherThread:循环处理事件队列,将事件派发给目标窗口。

·  关键线程分工

  • ·  android.display 线程(Java 层):处理 IMS 的消息(如配置变更、ANR 通知)。
  • InputReaderThread(Native 层):专注读取硬件事件,不阻塞其他操作。
  • InputDispatcherThread(Native 层):专注事件派发,确保实时性。

三、事件如何从硬件传到应用?

事件分发链:InputReader → InputDispatcher → 应用窗口

InputReader → InputDispatcher

    1. 通过QueuedInputListener将封装好的事件传递给InputDispatcher,存入mInboundQueue队列。

InputDispatcher 派发事件

    1. 从 WMS 获取焦点窗口信息(通过 IMS 同步),确定事件目标窗口。
    2. 通过InputChannel(跨进程通信通道)将事件发送给应用的InputConsumer,最终由ViewRootImpl处理并传递给界面组件(如按钮、文本框)。

相关文章:

  • 【R语言编程绘图-折线图】
  • inviteflood:基于 UDP 的 SIP/SDP 洪水攻击工具!全参数详细教程!Kali Linux教程!
  • WPF【09】WPF基础入门 (三层架构与MVC架构)
  • thinkadmin中使用layui日期选择器,数据库存储时间戳
  • 构建高可观测性的云原生应用体系:企业实践指南
  • Java爬虫,获取未来40天预测气象并写入Excel
  • 【HW系列】—目录扫描、口令爆破、远程RCE流量特征
  • Flink Table API 编程实战详解
  • 查询端口占用情况的命令(windows、linux)
  • 多语言实现插值查找算法
  • 一级菜单401问题
  • 【Linux】进程状态优先级
  • 国产三维CAD皇冠CAD(CrownCAD)建模教程:汽车电池
  • 《重塑认知:Django MVT架构的多维剖析与实践》
  • 将 ubutun 的网络模式 从NAT 改到 桥接模式后,无法上网,linux 没有IP地址 的解决方案
  • SQL Server 和 MySQL 对比
  • Maven工程演示
  • 数据仓库基础知识总结
  • DB2数据库HADR配置及详解
  • Jenkins分配对应项目权限与用户管理
  • 电子邮箱网站建设/朝阳seo排名优化培训
  • aspcms上传到虚拟主机后打开网站/个人推广app的妙招
  • 罗湖网站建设公司/推广app有哪些
  • 用微信公众平台做分类广告网站/结构优化
  • 公司网站内容相近/爱站工具包
  • 整个网站都在下雪特效怎么做/链接检测工具