當前位置 主頁 > 服務器問題 > Linux/apache問題 > 最大化 縮小

    Android 添加系統服務的方法詳解

    欄目:Linux/apache問題 時間:2020-02-06 08:25

    一、前言

    系統服務是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();
        }
      }
    }
    
    
    
    下一篇:沒有了
在线观看中文字幕理论片