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
繁體地址,請注明出處:http://hk.pswp.cn/diannao/84315.shtml
英文地址,請注明出處:http://en.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;其實也…