Android 添加系统服务的完整流程

[应用程序] (应用进程)│↓ 调用简单API
[SoundManager] │   ├─ 代理模式+门面模式(应用进程)│   ├─ 缓存数据 ←─ 装饰器模式(应用进程)│   └─ 转换异常 ←─ 适配器模式(应用进程)│↓ 通过Binder跨进程调用
[Binder Proxy]// 自动生成的Binder代理类(运行在应用进程,但处理跨进程通信)│↓ IPC(跨进程通信)
[SoundManagerService] (SystemServer进程)// 实际服务实现(运行在系统进程)└─ Binder Stub // 自动生成的Stub类(系统进程接收端)






以 Sound 服务为例,展示 Android 系统中添加一个系统服务的完整流程:

1. 定义 AIDL 接口

frameworks/base/core/java/android/media/ 创建接口文件:

// ISoundManager.aidl
package android.media;/** @hide */
interface ISoundManager {void playSound(in String soundName);void setVolume(int volume);int getCurrentVolume();boolean isMuted();
}

2. 实现服务端代码

系统服务的服务端实现(如 ActivityManagerService)通常放在 frameworks/base/services/

frameworks/base/services/core/java/com/android/server/sound/中实现服务:

// SoundManagerService.java
package com.android.server.sound;import android.content.Context;
import android.media.ISoundManager;public class SoundManagerService extends ISoundManager.Stub {private final Context mContext;private int mCurrentVolume = 50;private boolean mMuted = false;public SoundManagerService(Context context) {mContext = context;}@Overridepublic void playSound(String soundName) {enforceSoundPermission();// 实际播放声音的实现}@Overridepublic void setVolume(int volume) {mCurrentVolume = Math.max(0, Math.min(100, volume));}@Overridepublic boolean isMuted() {return mMuted;}private void enforceSoundPermission() {mContext.enforceCallingPermission("android.permission.MANAGE_SOUND","Sound permission required");}
}

3. 在 SystemServer 中启动服务

修改 frameworks/base/services/java/com/android/server/SystemServer.java

// 在 startOtherServices() 方法中添加
traceBeginAndSlog("StartSoundManagerService");
mSystemServiceManager.startService(SoundManagerService.class);
traceEnd();

4. 创建客户端管理器类

面向app开发的接口在frameworks/base/core/java/

存放的是 Android 框架的公共基础代码,主要分为两类:

1.面向应用开发者:提供 App 可调用的 SDK API(如 android.app、android.content)。
2.面向系统内部:实现 Android 核心机制(如 Binder、权限管理),这些对 App 透明,但支撑 App 运行。

frameworks/base/core/java/android/media/ 中创建:

// SoundManager.java
package android.media;import android.content.Context;public class SoundManager {public static final String SERVICE = "sound";private static ISoundManager sService;private final Context mContext;public static SoundManager get(Context context) {if (sService == null) {IBinder b = ServiceManager.getService(SERVICE);sService = ISoundManager.Stub.asInterface(b);}return new SoundManager(context);}private SoundManager(Context context) {mContext = context;}public void playSound(String soundName) {try {sService.playSound(soundName);} catch (RemoteException e) {throw e.rethrowFromSystemServer();}}public int getCurrentVolume() {try {return sService.getCurrentVolume();} catch (RemoteException e) {throw e.rethrowFromSystemServer();}}
}

5. 在 ContextImpl 中注册服务

修改 frameworks/base/core/java/android/app/ContextImpl.java

// 在 static {} 块中添加
registerService(SoundManager.SERVICE, new ServiceFetcher() {@Overridepublic Object createService(ContextImpl ctx) {return SoundManager.get(ctx);}
});

6. 添加权限声明

frameworks/base/core/res/AndroidManifest.xml 中添加:

<permission android:name="android.permission.MANAGE_SOUND"android:protectionLevel="signature" />

7. 更新系统配置

frameworks/base/core/res/res/values/config.xml 中添加:

<bool name="config_soundServiceEnabled">true</bool>

8. 编译和测试

编译系统并测试新服务:

bash
make -j8

9. 应用层调用示例

应用中使用 Sound 服务:

SoundManager soundManager = (SoundManager) getSystemService(Context.SOUND_SERVICE);
soundManager.playSound("notification");
int volume = soundManager.getCurrentVolume();

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/84315.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

wan2.1代码笔记

GPU内存不够&#xff0c;可以先运行umt5&#xff0c;然后再运行wanpipeline&#xff0c;参考FLUX.1代码笔记&#xff0c;或者使用ComfyUI。 下面使用随机数代替umt5 embedding。 import torch from diffusers.utils import export_to_video from diffusers import Autoencoder…

环境搭建与工具配置

3.1 本地环境搭建 3.1.1 WAMP环境搭建漏洞靶场&#xff08;一、二&#xff09; WAMP&#xff08;Windows Apache MySQL PHP&#xff09;是搭建本地Web漏洞靶场的基础环境。 安装步骤&#xff1a; Apache&#xff1a;下载并安装最新版Apache HTTP Server&#xff0c;配置监…

STM32F446主时钟失效时DAC输出异常现象解析与解决方案

—### 现象概述 在STM32F446微控制器应用中&#xff0c;若主时钟&#xff08;HSE&#xff09;的晶体信号对地短路&#xff0c;但DAC&#xff08;数模转换器&#xff09;仍能输出变化信号&#xff0c;这一现象看似矛盾&#xff0c;实则与系统时钟切换机制密切相关。本文将从硬件…

React 如何封装一个可复用的 Ant Design 组件

文章目录 前言一、为什么需要封装组件&#xff1f;二、 仿antd组件的Button按钮三、封装一个可复用的表格组件 (实战)1. 明确需求2. 设计组件 API3. 实现组件代码4. 使用组件 三、封装组件的最佳实践四、进阶优化 总结 前言 作为一名前端开发工程师&#xff0c;在日常项目中&a…

STC89C52RC/LE52RC

STC89C52RC 芯片手册原理图扩展版原理图 功能示例LED灯LED灯的常亮效果LED灯的闪烁LED灯的跑马灯效果&#xff1a;从左到右&#xff0c;从右到左 数码管静态数码管数码管计数mian.cApp.cApp.hCom.cCom.hDir.cDir.hInt.cInt.hMid.cMid.h 模板mian.cApp.cApp.hCom.cCom.hDir.cDir…

踩坑记录:RecyclerView 局部刷新notifyItemChanged多次调用只触发一次 onBindViewHolder 的原因

1. 问题背景 在做项目的时候&#xff0c;RecyclerView需要使用局部刷新&#xff0c;使用 notifyItemChanged(position, payload) 实现局部刷新&#xff0c;但发现调用多次只执行了一次&#xff0c;第二个刷新不生效。 2. 错误示例&#xff08;只处理 payloads.get(0)&#xff…

OpenLayers 加载鹰眼控件

注&#xff1a;当前使用的是 ol 5.3.0 版本&#xff0c;天地图使用的key请到天地图官网申请&#xff0c;并替换为自己的key 地图控件是一些用来与地图进行简单交互的工具&#xff0c;地图库预先封装好&#xff0c;可以供开发者直接使用。OpenLayers具有大部分常用的控件&#x…

WPF···

设置启动页 默认最后一个窗口关闭,程序退出,可以设置 修改窗体的icon图标 修改项目exe图标 双击项目名会看到代码 其他 在A窗体点击按钮打开B窗体,在B窗体设置WindowStartupLocation=“CenterOwner” 在A窗体的代码设置 B.Owner = this; B.Show(); B窗体生成在A窗体中间…

github公开项目爬取

import requestsdef search_github_repositories(keyword, tokenNone, languageNone, max_results1000):"""通过 GitHub API 搜索仓库&#xff0c;支持分页获取所有结果&#xff08;最多 1000 条&#xff09;:param keyword: 搜索关键词:param token: GitHub To…

防震基座在半导体晶圆制造设备抛光机详细应用案例-江苏泊苏系统集成有限公司

在半导体制造领域&#xff0c;晶圆抛光作为关键工序&#xff0c;对设备稳定性要求近乎苛刻。哪怕极其细微的振动&#xff0c;都可能对晶圆表面质量产生严重影响&#xff0c;进而左右芯片制造的成败。以下为您呈现一个防震基座在半导体晶圆制造设备抛光机上的经典应用案例。 企…

S32K开发环境搭建详细教程(一、S32K IDE安装注册)

一、S32K IDE安装注册 1、进入恩智浦官网https://www.nxp.com.cn/&#xff08;需要在官网注册一个账号&#xff09; 2、直接搜索 “Standard Software”&#xff0c;找到S32K3 Standard Software&#xff0c;点击进入 3、下载 (1)Automotive SW - S32K3 - S32 Design Studio…

Spring Cloud Gateway 微服务网关实战指南

上篇文章简单介绍了SpringCloud系列OpenFeign的基本用法以及Demo搭建&#xff08;Spring Cloud实战&#xff1a;OpenFeign远程调用与服务治理-CSDN博客&#xff09;&#xff0c;今天继续讲解下SpringCloud Gateway实战指南&#xff01;在分享之前继续回顾下本次SpringCloud的专…

MSP430G2553 USCI模块串口通信

1.前言 最近需要利用msp430连接蓝牙模块传递数据&#xff0c;于是死磕了一段时间串口&#xff0c;在这里记录一下 2.msp430串口模块 msp430的串口模块可以有USCI模块提供 在异步模式中&#xff0c; USCI_Ax 模块通过两个外部引脚&#xff0c; UCAxRXD 和 UCAxTXD&#xff0…

【产品经理从0到1】用户端产品设计与用户画像

思考 xx新闻的第一个版本应该做哪些事情呢&#xff1f; 用户端核心功能 用户端通用页面设计 思考 回想一下&#xff0c;大家在第一次使用一个新下载的App的时候会看到一些什么样的页面?这样的页面一般都是展示了一些什么内容? 引导页 概念 第一次安装App或者更新App后第…

多场景游戏AI新突破!Divide-Fuse-Conquer如何激发大模型“顿悟时刻“?

多场景游戏AI新突破&#xff01;Divide-Fuse-Conquer如何激发大模型"顿悟时刻"&#xff1f; 大语言模型在强化学习中偶现的"顿悟时刻"引人关注&#xff0c;但多场景游戏中训练不稳定、泛化能力差等问题亟待解决。Divide-Fuse-Conquer方法&#xff0c;通过…

佰力博科技与您探讨压电材料的原理与压电效应的应用

压电材料的原理基于正压电效应和逆压电效应&#xff0c;即机械能与电能之间的双向转换特性。 压电材料的原理源于其独特的晶体结构和电-机械耦合效应&#xff0c;具体可分为以下核心要点&#xff1a; 1. ‌正压电效应与逆压电效应的定义‌ ‌正压电效应‌&#xff1a;当压电…

算法备案审核周期

&#xff08;一&#xff09;主体备案审核 主体备案审核周期通常为7-10个工作日&#xff0c;监管部门将对企业提交的资质信息进行严格审查&#xff0c;审核重点包括&#xff1a; 营业执照的真实性、有效性及与备案主体的一致性。法人及算法安全责任人身份信息的准确性与有效性…

管理系统的接口文档

一、接口概述 本接口文档用于描述图书管理系统中的一系列 Restful 接口&#xff0c;涵盖图书的查询、添加、更新与删除操作&#xff0c;以及用户的登录注册等功能&#xff0c;方便客户端与服务器之间进行数据交互。 二、接口基础信息 接口地址&#xff1a;https://book-manag…

杰发科技AC7801——PWM获取固定脉冲个数

测试通道6 在初始化时候打开通道中断 void PWM1_GenerateFrequency(void) {PWM_CombineChConfig combineChConfig[1]; //组合模式相关结构体PWM_IndependentChConfig independentChConfig[2];//独立模式相关结构体PWM_ModulationConfigType pwmConfig; //PWM模式相关结构体PWM…

RL电路的响应

学完RC电路的响应&#xff0c;又过了一段时间了&#xff0c;想必很多人都忘了RC电路响应的一些内容。我们这次学习RL电路的响应&#xff0c;以此同时&#xff0c;其实也是带大家一起回忆一些之前所学的RC电路的响应的一些知识点。所以&#xff0c;这次的学习&#xff0c;其实也…