Android 16系统源码_无障碍辅助(一)认识无障碍服务
前言
Android 的无障碍辅助功能(Accessibility)是一套专为残障用户或特殊场景设计的核心技术框架,旨在让所有用户都能便捷地操作设备。其功能覆盖视觉、听觉、运动能力和认知障碍支持,同时为开发者提供标准化 API 以实现应用适配。
核心功能组件
Android 提供了多种内置无障碍工具和服务,涵盖不同场景需求:
功能名称 | 主要用途 | 适用人群 |
---|---|---|
TalkBack | 屏幕阅读器,通过语音和振动反馈界面内容 | 视障用户 |
Voice Access | 语音控制界面元素(可见即可说) | 运动障碍/免提操作 |
Switch Access | 通过物理开关或摄像头动作控制设备 | 重度运动障碍用户 |
Select to Speak | 点按或圈选文本后朗读内容 | 阅读障碍/学习困难用户 |
字幕和音频增强 | 实时字幕、声音放大器、左右声道平衡 | 听障用户 |
高对比度模式 | 调整界面颜色和字体大小以提升可读性 | 低视力用户 |
技术架构与原理
Android 无障碍功能的实现依赖多层系统服务协作。
事件传递机制
- 事件源:用户操作(如点击)或系统事件(如通知)触发 AccessibilityEvent。
- 收集与分发:系统级服务 AccessibilityManagerService 统一管理事件,按类型分发给注册的无障碍服务。
- 处理响应:开发者实现的 AccessibilityService 通过 onAccessibilityEvent() 接收事件并执行逻辑(如点击按钮、读取文本)。
界面元素访问
- 控件树解析:通过 AccessibilityNodeInfo 获取界面元素的层级结构、属性(如文本、坐标)。
- 窗口快照:AccessibilityWindowManager 维护窗口状态,提供跨应用的界面内容访问(需用户授权)。
语音与交互
- 语音合成(TTS):TalkBack 使用 TextToSpeech 引擎朗读文本。
- 手势映射:自定义手势(如双指滑动)触发操作,由 AccessibilityGestureDetector 解析。
简单案例
下面我们将会编写一个简单的案例,具体认识一下 Android 无障碍服务 (AccessibilityService) 。此案例主要用于获取当前屏幕上的所有文本信息。
步骤 1:添加无障碍服务声明(AndroidManifest.xml )
<serviceandroid:name=".MyAccessibilityService"android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"android:exported="true"><intent-filter><action android:name="android.accessibilityservice.AccessibilityService" /></intent-filter><meta-dataandroid:name="android.accessibilityservice"android:resource="@xml/accessibility_service_config" />
</service>
步骤 2:创建无障碍服务配置文件(res/xml/accessibility_service_config.xml)
<accessibility-servicexmlns:android="http://schemas.android.com/apk/res/android"android:description="@string/accessibility_service_desc"android:accessibilityEventTypes="typeWindowContentChanged|typeWindowStateChanged"android:accessibilityFlags="flagDefault|flagRetrieveInteractiveWindows"android:canRetrieveWindowContent="true"android:settingsActivity="com.example.MainActivity"/>
步骤 3:实现无障碍服务类
public class MyAccessibilityService extends AccessibilityService {@Overridepublic void onServiceConnected() {Log.d("Accessibility", "无障碍服务已启动");}@Overridepublic void onAccessibilityEvent(AccessibilityEvent event) {int eventType = event.getEventType();if (eventType == AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED ||eventType == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) {AccessibilityNodeInfo rootNode = getRootInActiveWindow();if (rootNode == null) return;List<String> texts = getAllTexts(rootNode);Log.d("ScreenTexts", "捕获文本: " + TextUtils.join(", ", texts)); sendTextToActivity(texts);rootNode.recycle(); // 释放资源}}private List<String> getAllTexts(AccessibilityNodeInfo node) {List<String> texts = new ArrayList<>();traverseNode(node, texts);return filterUniqueTexts(texts);}private void traverseNode(AccessibilityNodeInfo node, List<String> texts) {if (node == null) return;CharSequence text = node.getText();if (text != null && !TextUtils.isEmpty(text)) {texts.add(text.toString().trim());}for (int i = 0; i < node.getChildCount(); i++) {AccessibilityNodeInfo child = node.getChild(i);if (child != null) {traverseNode(child, texts);child.recycle(); // 必须回收子节点}}}private List<String> filterUniqueTexts(List<String> list) {Set<String> uniqueSet = new HashSet<>(list);return new ArrayList<>(uniqueSet);}private void sendTextToActivity(List<String> texts) {Intent intent = new Intent("ACTION_UPDATE_TEXT");intent.putStringArrayListExtra("texts", new ArrayList<>(texts));sendBroadcast(intent);}@Overridepublic void onInterrupt() {}
}
启用无障碍服务
需要在系统设置中启用应用的无障碍服务。
运行效果
当界面内容变化时(如打开新页面),服务会自动捕获所有可见文本,并打印出来。
在我们在设置页面开启无障碍服务的时候
TextAccessibilityService com.example.myapplication I 无障碍服务已启动
TextAccessibilityService com.example.myapplication I 捕获文本: “TextAccessibility_tip”快捷方式, 关闭, 设置, 选项, 使用“TextAccessibility_tip”, TextAccessibilityDesc, TextAccessibility_tip
打开首页
TextAccessibilityService com.example.myapplication I 捕获文本: 设置, 文件, 时钟, 拨打电话, MyApplication, WebView Browser Tester, 相机, 日历, TMoble, 通讯录, Search, 搜索应用, 短信, 图库
TextAccessibilityService com.example.myapplication I 捕获文本: 拨打电话, 短信, WebView Browser Tester, 相机, 图库
打开MainActivity
TextAccessibilityService com.example.myapplication I 捕获文本: Hello World!
注意事项
权限限制
-
Android 11+ 需要 android.permission.QUERY_ALL_PACKAGES 权限才能获取其他应用的内容。
-
敏感场景(如密码输入)可能无法捕获文本。
性能优化
-
避免频繁遍历控件树(可增加事件过滤)。
-
使用 postDelayed 防抖处理高频事件。
相关文章:

Android 16系统源码_无障碍辅助(一)认识无障碍服务
前言 Android 的无障碍辅助功能(Accessibility)是一套专为残障用户或特殊场景设计的核心技术框架,旨在让所有用户都能便捷地操作设备。其功能覆盖视觉、听觉、运动能力和认知障碍支持,同时为开发者提供标准化 API 以实现应用适配…...

分布式数据库备份实践
在分布式备份中可以采取两种方式进行备份,一种是采用手动编写backup.yml文件进行备份,另外一种是吧备份过程交给备份工具自动执行。如果需要个性化进行备份,建议采用手动编写备份文件方式进行备份。 以下是针对两种备份方式的实践:…...

如何发布npm包?
如何发布npm包? 1. 注册账号[npm官网](https://www.npmjs.com/)2. 检查 npm 源是否在官方 npm 仓库,如果不在,进行切换3. 检查4. 打包配置5. 发布6. 使用错误:版本更新命令 1. 注册账号npm官网 2. 检查 npm 源是否在官方 npm 仓库…...
鸿蒙---使用真机模拟器的时候,图片不加载问题
使用真机模拟器的时候,图片不加载问题 解决方案: 1,找到 module.json5 文件,路径 entry -> src -> main -> module.json5 2,在module.json5 文件中,开头的’module’中添加 "requestPermiss…...

实验设计与分析(第6版,Montgomery)第5章析因设计引导5.7节思考题5.6 R语言解题
本文是实验设计与分析(第6版,Montgomery著,傅珏生译) 第5章析因设计引导5.7节思考题5.6 R语言解题。主要涉及方差分析,正态假设检验,残差分析,交互作用图,等值线图。 dataframe <-data.frame…...
.NET 8使用AOT发布ASP.NET Core应用
.NET 8 使用 AOT 发布 ASP.NET Core 应用 一、AOT 编译简介 在 .NET 开发中,编译方式有多种,其中 AOT(Ahead-Of-Time)编译是一种区别于传统 JIT(Just-In-Time)编译的方式。传统的 JIT 编译是在 .NET 应用…...

OpenCV计算机视觉实战(8)——图像滤波详解
OpenCV计算机视觉实战(8)——图像滤波详解 0. 前言1. 线性滤波1.1 均值滤波1.2 高斯滤波1.3 拉普拉斯滤波1.4 Sobel 滤波 2. 非线性滤波3. 自定义卷积核小结系列链接 0. 前言 在本文中,我们将深入探索线性与非线性滤波的算法原理、性能优化及…...
Docker 前端镜像容器部署指南
1. 编写 Dockerfile 文件 # 使用轻量级的 Nginx Alpine 作为基础镜像 FROM nginx:alpine# 设置工作目录 WORKDIR /usr/share/nginx/html# 删除默认的 Nginx 静态文件 RUN rm -rf ./*# 复制本地 dist 目录下的文件到容器中 COPY ./dist /usr/share/nginx/html# 暴露容器端口 EX…...
OpenAI大模型不听人类指令事件的技术分析与安全影响
OpenAI大模型不听人类指令事件的技术分析与安全影响 OpenAI大模型o3确实存在不遵从人类关闭指令的现象,这一行为已被第三方安全机构验证,但其本质是技术缺陷而非AI意识觉醒。帕利塞德研究所的测试显示,在100次实验中o3有7次成功绕过关闭指令…...
图神经网络实战——图的可视化
图神经网络实战——图的可视化 0. 前言1. networkx2. Gephi相关链接0. 前言 图结构可通过图形化方式直观呈现。节点通常用圆形表示,边则用连接线表示。 然而当节点和边数量增加时,绘制清晰的图形表示可能会变得相当困难,这主要源于节点在二维坐标系中的布局问题。对于包含数…...

自动化安全脚本学习
1.目录扫描器 目标:使用python编写一个自动化目录扫描工具,实现简单信息收集,判断目标网站是否存在常见路径。 import requests #用于发HTTP请求 from concurrent.futures import ThreadPoolExecutor #实现多线程扫描# 扫描目标 target h…...
github公开项目爬取
import requestsdef search_github_repositories(keyword, tokenNone, languageNone, max_results1000):"""通过 GitHub API 搜索仓库,支持分页获取所有结果(最多 1000 条):param keyword: 搜索关键词:param token: GitHub To…...
用豆包写单元测试
用豆包写单元测试, 输入 vue 模板内容,输入 参考vue模板内容写一个单元测试要求用jest.mock实现构造完成,修复bug。npm run test:unit – tests/unit/views/xxx/xxx.spec.js看下 % Stmts 语句覆盖率:执行到的代码语句占总语句的比…...

传输层协议TCP(上)
上一篇https://blog.csdn.net/Small_entreprene/article/details/148143494?fromshareblogdetail&sharetypeblogdetail&sharerId148143494&sharereferPC&sharesourceSmall_entreprene&sharefromfrom_link 上文学习了传输层的协议之一UDP,接下来…...

Windows下安装并使用kubectl查看K8S日志
【1】安装kubectl 官网文档:https://kubernetes.io/zh-cn/docs/tasks/tools/install-kubectl-windows/ 下载后得到 kubectl.exe,放到一个目录下,然后配置环境变量。 此时CMD 进入DOS命令窗口 kubectl version【2】配置config文件 其实就是…...
Hive 分区详解:从基础概念到实战应用
一、为什么要分区? 1.将大规模数据按规则(如时间、地域)划分到不同目录,提升数据组织性。 2.通过分区过滤,减少扫描数据量,显著提升查询效率。 3.不同分区可对应不同业务线或权限,增强数据隔…...

Android studio进阶开发(六)--如何用真机通过okhttp连接服务器
我们学过了如何通过okhttp查询网络上已经发布的网页,但我们还需要在做全栈时保证前后端能够交互。 前要课程 okhttp的使用 真机端口连接 安全认证 由于http的安全性较差,在没有安全协议的情况下,使用自己的后端连接会报错,所以…...
如何解决网站服务器的异常问题?
当网站服务器出现异常情况,导致用户无法正常访问网页信息的时候,该如何解决这一问题呢?小编下面就带领大家共同探讨一下这一问题。 企业在面对网站服务器异常时,首先要对服务器硬件设备进行详细的检查,可以使用硬盘检测…...

WeakAuras Lua Script [ICC BOSS 11 - Sindragosa]
WeakAuras Lua Script [ICC BOSS 11 - Sindragosa] 冰冠堡垒Icecrown Citadel 冰龙 辛达苟萨(寒冰信标插件) 左 (绿,黄) 中(蓝,紫) 右(白,橙) lua script&…...
用户界面禁忌——基础原则
文章目录 基本原则1:关注用户及其任务,而不是技术基本原则2:首先考虑功能,然后才是表示基本原则3:与用户对任务的看法保持一致基本原则4:设计要符合常见情况基本原则5:不要分散用户对他们目标的…...

电脑开机后出现bootmgr is conmpressed原因及解决方法
最近有网友问我为什么我电脑开机后出现BOOTMGR is compressed,这个提示意思是:意思是启动管理器被压缩了,即使重启也无法正常进入系统。原因有很多,大部分是引导出现问题,或选错了启动硬盘所导致的,下面我们来详细分析…...

vite配置一个css插件
vite.config.js的plugins执行函数 该例子只是替换一些css,具体内容不重要,主要看形参的运用 // vite-plugin-css.js export default function cssPlugin() {return {name: vite-plugin-css-post, // 插件的名字,Vite 插件必须有名字enforce: post, // 设定插件执…...

React+Taro 微信小程序做一个页面,背景图需贴手机屏幕最上边覆盖展示
话不多说 直接上图 第一步 import { getSystemInfoSync } from tarojs/taro;第二步 render() {const cardBanner getImageUrlByGlobal(member-merge-bg.png);const { safeArea, statusBarHeight } getSystemInfoSync();const NAV_BAR_HEIGHT 44;const navBarHeight NAV…...

Spring框架学习day4--Spring集成Mybatis(IOC)
Spring集成Mybatis1.添加jar包(pom.xml)2.配置sqlSessionFactiory(spring.xml)3.再service类中注入Dao代理接口4.测试类5文件结构 Spring集成Mybatis Spring集成Mybatis其核心是将SqlSessionFactory交由Spring管理,并由 Spring管理…...
太阳系运行模拟程序-html动画
太阳系运行模拟程序-html动画 by AI: <!DOCTYPE html> <html lang"zh"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>交互式太阳系…...

【C++ Qt】容器类(GroupBox、TabWidget)内附思维导图 通俗易懂
每日激励:“不设限和自我肯定的心态:I can do all things。 — Stephen Curry” ✍️绪论: 本章主要介绍了 Qt 中 QGroupBox 与 QTabWidget 控件。QGroupBox 是带标题的分组框,能容纳其他控件,有标题、对齐方式、是否…...

SOC-ESP32S3部分:18-串口
飞书文档https://x509p6c8to.feishu.cn/wiki/NqrMw6X8Si6sSqkyPbxcFRxGnid UART全称是通用异步接收器/发送器,ESP32-S3 芯片有 3 个 UART 控制器。每个 UART 控制器可以独立配置波特率、数据位长度、位顺序、停止位位数、奇偶校验位等参数。 串口文档参考…...
CSS 样式表的四种应用方式及css注释的应用小结
CSS样式表的四种应用方式及注释应用小结 一、样式表应用方式 内联样式(行内样式) <div style"color: #ff0000; font-size: 16px;">示例文本</div>特点:直接写在HTML标签的style属性中优先级:最高ÿ…...
五、web安全--XSS漏洞(2)--XSS相关payload
XSS 主要是针对网页客户端的一种攻击,那么就要执行 JavaScript 代码,那么无疑需要用到 JavaScript 语言以及在 HTML 中可以解析 JavaScript 代码的标签。 1、标签类 (1)script <script></script>标签是最直接的 xS…...
AI架构师的新工具箱:ChatGPT、Copilot、AutoML、模型服务平台
AI架构师不仅要懂架构、懂AI服务,还需要具备使用AI工具提升工作效率的能力。新一代AI工具已经成为架构师不可或缺的“工具箱”,帮助他们更高效地进行设计、部署、编码与优化。 以下介绍几类代表性工具,并说明它们在实际架构工作中的应用场景。 一、ChatGPT:生成架构设计文…...