一、前言
系統(tǒng)服務(wù)是Android中非常重要的一部分, 像ActivityManagerService, PackageManagerService, WindowManagerService, 這些系統(tǒng)服務(wù)都是Framework層的關(guān)鍵服務(wù), 本篇文章主要講一下如何基于Android源碼添加一個(gè)系統(tǒng)服務(wù)的完整流程, 除了添加基本系統(tǒng)服務(wù), 其中還包含添加JNI部分代碼和App通過(guò)AIDL調(diào)用的演示Demo, 調(diào)用包含App調(diào)用服務(wù)端, 也包含服務(wù)端回調(diào)App, 也就是完成一個(gè)簡(jiǎn)單的雙向通信.
注: 測(cè)試代碼基于Android 7.1.1, 其他Android版本都是大同小異.
二、編寫AIDL文件
添加服務(wù)首先是編寫AIDL文件, AIDL文件路徑如下:
frameworks/base/core/java/com/example/utils/
1.ISystemEvent.aidl 內(nèi)容如下:
package com.example.utils;import com.example.utils.IEventCallback;interface ISystemEvent { void registerCallback(IEventCallback callback); void unregisterCallback(IEventCallback callback); void sendEvent(int type, String value);}
2.IEventCallback.aidl 內(nèi)容如下
package com.example.utils;interface IEventCallback{ oneway void onSystemEvent(int type, String value);}
AIDL文件編寫, 教程很多, 我這里就不詳細(xì)說(shuō)明了, 需要注意的是, 由于我們要實(shí)現(xiàn)回調(diào)功能, 所以必須寫一個(gè)回調(diào)接口 IEventCallback, 另外AIDL文件中 oneway 關(guān)鍵字表明調(diào)用此函數(shù)不會(huì)阻塞當(dāng)前線程, 調(diào)用端調(diào)用此函數(shù)會(huì)立即返回, 接收端收到函數(shù)調(diào)用是在Binder線程池中的某個(gè)線程中. 可以根據(jù)實(shí)際項(xiàng)目需求選擇是否需要加 oneway 關(guān)鍵字.
AIDL只支持傳輸基本java類型數(shù)據(jù), 要想傳遞自定義類, 類需要實(shí)現(xiàn) Parcelable 接口, 另外, 如果傳遞基本類型數(shù)組, 需要指定 in out 關(guān)鍵字, 比如 void test(in byte[] input, out byte[] output) , 用 in 還是 out, 只需要記住: 數(shù)組如果作為參數(shù), 通過(guò)調(diào)用端傳給被調(diào)端, 則使用 in, 如果數(shù)組只是用來(lái)接受數(shù)據(jù), 實(shí)際數(shù)據(jù)是由被調(diào)用端來(lái)填充的, 則使用 out, 這里之所以沒(méi)有說(shuō)服務(wù)端和客戶端, 是因?yàn)?in out 關(guān)鍵字用哪個(gè)和是服務(wù)端還是客戶端沒(méi)有聯(lián)系, 遠(yuǎn)程調(diào)用和被調(diào)用更適合描述.
文件寫完后, 添加到編譯的 Android.mk 中 LOCAL_SRC_FILES 后面:
3.frameworks/base/Android.mk
LOCAL_SRC_FILES += / core/java/android/view/IWindow.aidl / core/java/android/view/IWindowFocusObserver.aidl / core/java/android/view/IWindowId.aidl / 部分代碼省略 ... core/java/com/example/utils/ISystemEvent.aidl / core/java/com/example/utils/IEventCallback.aidl / 部分代碼省略 ...
編譯代碼, 編譯前需執(zhí)行 make update-api, 更新接口, 然后編譯代碼,確保AIDL編寫沒(méi)有錯(cuò)誤, 編譯后會(huì)生成對(duì)應(yīng)java文件, 服務(wù)端要實(shí)現(xiàn)對(duì)應(yīng)接口.
三、編寫Manager類
我們可以看到, Android API 中有很多Manager類, 這些類一般都是某個(gè)系統(tǒng)服務(wù)的客戶端代理類, 其實(shí)我們不寫Manager類, 只通過(guò)AIDL文件自動(dòng)生成的類, 也可以完成功能, 但封裝一下AIDL接口使用起來(lái)更方便, 我們測(cè)試用的Manager類為 SystemEventManager, 代碼如下:
frameworks/base/core/java/com/example/utils/SystemEventManager.java
package com.example.utils;import android.content.Context;import android.os.RemoteException;import android.util.Log;import com.example.example.ISystemEvent;import com.example.IEventCallback;public class SystemEventManager { private static final String TAG = SystemEventManager.class.getSimpleName(); // 系統(tǒng)服務(wù)注冊(cè)時(shí)使用的名字, 確保和已有的服務(wù)名字不沖突 public static final String SERVICE = "test_systemevent"; private final Context mContext; private final ISystemEvent mService; public SystemEventManager(Context context, ISystemEvent service) { mContext = context; mService = service; Log.d(TAG, "SystemEventManager init"); } public void register(IEventCallback callback) { try { mService.registerCallback(callback); } catch (RemoteException e) { Log.w(TAG, "remote exception happen"); e.printStackTrace(); } } public void unregister(IEventCallback callback) { try { mService.unregisterCallback(callback); } catch (RemoteException e) { Log.w(TAG, "remote exception happen"); e.printStackTrace(); } } /** * Send event to SystemEventService. */ public void sendEvent(int type, String value) { try { mService.sendEvent(type, value); } catch (RemoteException e) { Log.w(TAG, "remote exception happen"); e.printStackTrace(); } }}
新聞熱點(diǎn)
疑難解答
圖片精選