一、前言
系統服務是Android中非常重要的一部分, 像ActivityManagerService, PackageManagerService, WindowManagerService, 這些系統服務都是Framework層的關鍵服務, 本篇文章主要講一下如何基于Android源碼添加一個系統服務的完整流程, 除了添加基本系統服務, 其中還包含添加JNI部分代碼和App通過AIDL調用的演示Demo, 調用包含App調用服務端, 也包含服務端回調App, 也就是完成一個簡單的雙向通信.
注: 測試代碼基于Android 7.1.1, 其他Android版本都是大同小異.
二、編寫AIDL文件
添加服務首先是編寫AIDL文件, AIDL文件路徑如下:
frameworks/base/core/java/com/example/utils/
1.ISystemEvent.aidl 內容如下:
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 內容如下
package com.example.utils;interface IEventCallback{ oneway void onSystemEvent(int type, String value);}
AIDL文件編寫, 教程很多, 我這里就不詳細說明了, 需要注意的是, 由于我們要實現回調功能, 所以必須寫一個回調接口 IEventCallback, 另外AIDL文件中 oneway 關鍵字表明調用此函數不會阻塞當前線程, 調用端調用此函數會立即返回, 接收端收到函數調用是在Binder線程池中的某個線程中. 可以根據實際項目需求選擇是否需要加 oneway 關鍵字.
AIDL只支持傳輸基本java類型數據, 要想傳遞自定義類, 類需要實現 Parcelable 接口, 另外, 如果傳遞基本類型數組, 需要指定 in out 關鍵字, 比如 void test(in byte[] input, out byte[] output) , 用 in 還是 out, 只需要記住: 數組如果作為參數, 通過調用端傳給被調端, 則使用 in, 如果數組只是用來接受數據, 實際數據是由被調用端來填充的, 則使用 out, 這里之所以沒有說服務端和客戶端, 是因為 in out 關鍵字用哪個和是服務端還是客戶端沒有聯系, 遠程調用和被調用更適合描述.
文件寫完后, 添加到編譯的 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 / 部分代碼省略 ...
編譯代碼, 編譯前需執行 make update-api, 更新接口, 然后編譯代碼,確保AIDL編寫沒有錯誤, 編譯后會生成對應java文件, 服務端要實現對應接口.
三、編寫Manager類
我們可以看到, Android API 中有很多Manager類, 這些類一般都是某個系統服務的客戶端代理類, 其實我們不寫Manager類, 只通過AIDL文件自動生成的類, 也可以完成功能, 但封裝一下AIDL接口使用起來更方便, 我們測試用的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(); // 系統服務注冊時使用的名字, 確保和已有的服務名字不沖突 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(); } }}
新聞熱點
疑難解答
圖片精選