本文介紹了Android進程間通信實踐的示例代碼,分享給大家,具體如下:

因為線程間的內存是共享的,所以它們之間的通信簡單,比如可以通過共享變量等方式實現。而進程間想要通信就要麻煩許多了。要想實現進程間通信,我們需要在不同進程之間定義一套它們可以共同理解的接口描述語言,也即 IDL。比較常用的 IDL 有 JSON、Protocol Buffers 等。而 Android 不同進程之間的通信也有個特別的語言,叫 AIDL(Android Interface Definition Language)。下面我們就來用一個場景說明下,我們如何使用這個 AIDL。
場景:現在有個叫 Teacher 應用,和多個 Student 應用。Teacher 應用先安裝在手機上,當某個 Student 應用安裝時,就會把自己的包名和進程號告訴 Teacher 應用,Teacher 應用會把這些信息打印出來,并根據這些信息計算出一個別名傳回給 Student,而 Student 應用獲得別名后會展示出來。
定義協議
首先我們在 Teacher 的工程中創建一個叫 rule 的 module 來定義通信規則。我們在 rule module 中創建一個 ITeacher.aidl 文件(建議直接使用 Android Studio 中的 New 方式創建 aidl )。內容如下:
// ITeacher.aidlpackage me.pengtao.rule;import me.pengtao.rule.StudentInfo;interface ITeacher { // 向老師注冊自己的Id void registerId(in String id); // 學生獲得自己的昵稱 String getAlias();}該文件需在與 java 同級的 aidl 目錄中,其中方法的參數可以為基本類型(int、float、boolean、double、String),其它自定義類型需要實現 Parcelable 接口才可使用,所以那些包裝類因為實現的是Serializable 接口,所以無法使用。
創建完 aidl 文件后,build 一下項目后,則會在 build/generated/source/aidl 目錄下,生成一個與該 aidl 文件同名的 java 類。證明 aidl 創建成功。
注:此處之所以新創建一個 module 來定義 AIDL,是因為不同的進程間只能通過相同命稱和包名的 AIDL 進行通信。此處創建一個 module 后,后續可以做成依賴分發給需要和本進程通信的APP。
另外細心的讀者可能也看到了 AIDL 的方法參數前面有個 in 關鍵字,在 AIDL語法中針對參數的流方向有以下三個關鍵字定義。
in:表示 Client 向 Server 輸入;
out:表示 Client 從 Server 端獲取數據;
inout:表示既可獲取,也可輸入;
有關 Client 和 Server 的解釋下面有詳細說明。
使用這些關鍵字時需要注意以下兩點:
注冊 Service
AIDL的進程通信類似一個 Server / Client 模型,一邊作為 Server 負責接收和分發數據,另一端作為 Client 端,負責連接到 Server 端并進行數據交互。此處的 Teacher 充當 Server 端。所以此處我們需要為 Teacher 提供一個 Service 來向其它 Client 提供服務,此 Service 的實現如下:
public class TeacherService extends Service { private String mId; @Override public IBinder onBind(Intent intent) { return mBinder; } IBinder mBinder = new ITeacher.Stub() { @Override public void registerId(final String id) throws RemoteException { mId = id; } @Override public String getAlias() throws RemoteException { return mId + "Alias"; } };}<permission android:name="me.pengtao.teacher.RULE"/><service android:name=".TeacherService" android:exported="true" android:permission="me.pengtao.teacher.RULE" />
完成以上步驟后,我們運行 Teacher。然后我們再來創建一個 StudentA 的 project ,該工程依賴 rule,其中關鍵代碼實現如下:
private ITeacher mTeacher; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Intent serviceIntent = new Intent().setComponent(new ComponentName("me.pengtao.teacher", "me.pengtao.teacher.TeacherService")); bindService(serviceIntent, mConnection, BIND_AUTO_CREATE); } ServiceConnection mConnection = new ServiceConnection() { @Override public void onServiceConnected(final ComponentName name, final IBinder service) { mTeacher = ITeacher.Stub.asInterface(service); try { mTeacher.registerId(BuildConfig.APPLICATION_ID); ((TextView) findViewById(R.id.alias)).setText(getString(R.string.alias, mTeacher .getAlias())); } catch (RemoteException e) { e.printStackTrace(); } } @Override public void onServiceDisconnected(final ComponentName name) { mTeacher = null; } };運行 StudentA 后,它首先進行 TeacherService 的綁定操作,綁定成功后,把自己的包名發送給 Teacher,然后 Teacher 返回一個別名,并展示出來。
POJO類
以上我們展示了基本類型的通信方式,此處我們再來說下如何在不同進程間傳輸一個POJO類。比如現在老師不是只要求學生提供一個 ID,而是需要他們提供一套信息,比如自己的名字,性別等。那要如何去做呢?
首先我們需要在 rule 的 module 的 java 目錄下定義這樣一個pojo 類并 implements Parcelable。然后我們再在 aidl 目錄下創建一個對應名稱和包名的 aidl 文件。內容如下:
// IStudentInfo.aidlpackage me.pengtao.rule;parcelable StudentInfo;
這樣就大功告成。但在使用POJO類時,需注意以下幾點:
1,aidl的命名必須與POJO類命名一樣,否則會出現以下錯誤;
Error:Execution failed for task ':rule:compileDebugAidl'.> java.io.IOException: com.android.ide.common.process.ProcessException: Error while executing process /Users/chris/Library/Android/sdk/build-tools/26.0.2/aidl with arguments {-p/Users/chris/Library/Android/sdk/platforms/android-26/framework.aidl -o/Users/chris/kujiale/Teacher/rule/build/generated/source/aidl/debug -I/Users/chris/kujiale/Teacher/rule/src/debug/aidl -I/Users/chris/kujiale/Teacher/rule/src/main/aidl -I/Users/chris/.gradle/caches/transforms-1/files-1.1/support-media-compat-26.1.0.aar/a2b4d6b89d0b463dae37ed4ed6eac440/aidl -I/Users/chris/.gradle/caches/transforms-1/files-1.1/support-compat-26.1.0.aar/80390967c8d1c220862dcd091cfa81ca/aidl -d/var/folders/t9/x4yt536976l_sd9xcn2d6xwm0000gn/T/aidl5092216096502271751.d /Users/chris/kujiale/Teacher/rule/src/main/aidl/me/pengtao/rule/ITeacherWant.aidl}2,POJO類中必須要實現一個默認的構造函數,否則會報以下錯誤;
Error:(49, 9) error: constructor StudentInfo in class StudentInfo cannot be applied to given types;required: Parcelfound: no argumentsreason: actual and formal argument lists differ in length
3,POJO類必須使用 Parcelable 方式,并且其中必須實現一個 readFromParcel(Parcel) 方法,否則會報以下錯誤;
Error:(88, 5) error: cannot find symbol method readFromParcel(Parcel)
4,POJO類作為參數時,前面一定需要用in、out或inout修飾。關于這三個關鍵字的用法,可見前面內容。
Demo代碼:
https://github.com/CPPAlien/AIDL_example_Teacher
https://github.com/CPPAlien/AIDL_example_Student
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。
新聞熱點
疑難解答