【轉載】Android Studio Service AIDL 詳解

公司產品以前IM這塊存在不少問題,消息到達率低,加上協議上有些問題,丟消息頻繁,因此須要重構IM,AIDL不能解決以上問題。好吧!那AIDL能夠解決什麼問題?什麼是AIDL?php

什麼是AIDL?

AIDL是 Android Interface definition language的縮寫,它是一種Android內部進程通訊接口的描述語言,經過它咱們能夠定義進程間的通訊接口java

AIDL能夠解決什麼問題?

  • 能夠實現多個應用程序共享同一個Service的功能,好比:IM服務能夠提供給多個APP使用,先在推送基本都是採起這種方案
  • 能夠跨進程調用服務裏的方法

Android Studio AIDL 實戰

大部分文章介紹都是在eclipse下介紹的,如今 Android Studio 做爲開發工具比較普及了,因此我在Android Studio 下介紹(其實區別不大)。android

言歸正傳,咱們須要使用Android Studio實現一個遠程Service,而且創建AIDL進行通訊。git

搭建了簡單的Service框架

1.繼承Servicegithub

package name.quanke.aidldemo;

import android.app.Service; import android.content.Intent; import android.os.IBinder; /** * * Created by http://quanke.name on 16/7/23. */ public class PushService extends Service { public PushService() { } @Override public IBinder onBind(Intent intent) { return new LibHandler(); } @Override public boolean onUnbind(Intent intent) { return super.onUnbind(intent); } }

2.在AndroidManifest.xml裏註冊sql

<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="name.quanke.aidldemo"> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:name=".App" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN"/> <category android:name="android.intent.category.LAUNCHER"/> </intent-filter> </activity> <service android:name=".PushService" android:enabled="true" android:process=":push" android:exported="true"> </service> </application> </manifest>

創建AIDL

1.建立AIDL文件夾

1.建立AIDL文件夾

1.建立AIDL文件夾ruby

2.建立AIDL文件

2.建立AIDL文件

2.建立AIDL文件服務器

3.編寫AIDL文件
// IHandler.aidl
package name.quanke.aidldemo; // Declare any non-default types here with import statements interface IHandler { void connect(); }
4.AIDL文件 生成接口

4.AIDL文件 生成接口

4.AIDL文件 生成接口app

生成後的樣子框架

4.AIDL文件 生成後的樣子

4.AIDL文件 生成後的樣子

5.編寫客戶端 ServiceConnection
package name.quanke.aidldemo;

import android.app.Application; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.ServiceConnection; import android.os.IBinder; import android.os.RemoteException; import android.util.Log; /** * Created by quanke on 16/7/23. */ public class PushManager { private static final String TAG = "PushManager.class"; private IHandler iHandler; private static PushManager ourInstance = new PushManager(); public static PushManager getInstance() { return ourInstance; } private PushManager() { } public void init(Application app){ Intent binderIntent = new Intent(app,PushService.class); app.bindService(binderIntent, serviceConnection, Context.BIND_AUTO_CREATE); } public void connect(){ try { //經過AIDL遠程調用 Log.d(TAG,"++start Remote++"); iHandler.connect(); } catch (RemoteException e) { e.printStackTrace(); } } private ServiceConnection serviceConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { iHandler = IHandler.Stub.asInterface(service); //鏈接成功調動 } @Override public void onServiceDisconnected(ComponentName name) { //斷開鏈接調用 } }; }
6.編寫服務端實現connect方法
package name.quanke.aidldemo;

import android.os.IBinder; import android.os.RemoteException; import android.util.Log; /** * * Created by quanke on 16/7/23. */ public class LibHandler extends IHandler.Stub{ @Override public void connect() throws RemoteException { Log.d("","connect()"); } @Override public IBinder asBinder() { return null; } }

以上實現了簡單的鏈接,接下來咱們實現傳遞自定義類型

傳遞自定義的類型

AIDL默認支持的類型包括Java基本類型(int、long、boolean等),和(String、List、Map、CharSequence),若是要傳遞自定義的類型須要實現android.os.Parcelable接口。本身寫了一個實體類public class Message implements Parcelable。

package name.quanke.aidldemo.model;

import android.os.Parcel; import android.os.Parcelable; /** * * Created by quanke on 16/7/23. */ public class Message implements Parcelable { private long id; private String content; public long getId() { return id; } public void setId(long id) { this.id = id; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } @Override public String toString() { return "Message{" + "id=" + id + ", content='" + content + '\'' + '}'; } @Override public int describeContents() { return 0; } @Override public void writeToParcel(Parcel dest, int flags) { dest.writeLong(this.id); dest.writeString(this.content); } public Message() { } protected Message(Parcel in) { this.id = in.readLong(); this.content = in.readString(); } public static final Creator<Message> CREATOR = new Creator<Message>() { @Override public Message createFromParcel(Parcel source) { return new Message(source); } @Override public Message[] newArray(int size) { return new Message[size]; } }; }

修改IHandler

// IHandler.aidl
package name.quanke.aidldemo; // Declare any non-default types here with import statements import name.quanke.aidldemo.model.Message; interface IHandler { void connect(); void sendMessage(Message message); }

編譯項目,報錯

/Users/quanke/Dev/android/src/AIDLDemo/app/src/main/aidl/name/quanke/aidldemo/IHandler.aidl
Error:(6) couldn't find import for class name.quanke.aidldemo.model.Message Error:Execution failed for task ':app:compileDebugAidl'. > java.lang.RuntimeException: com.android.ide.common.process.ProcessException: org.gradle.process.internal.ExecException: Process 'command '/Users/quanke/Dev/android/tools/android-sdks/build-tools/23.0.3/aidl'' finished with non-zero exit value 1 Information:BUILD FAILED

由於自定義類型不只要定義實現android.os.Parcelable接口的類,還得爲該實現類定義一個aidl文件,以下:

自定義類aidl文件

自定義類aidl文件

// IHandler.aidl
package name.quanke.aidldemo.model;

// Declare any non-default types here with import statements import name.quanke.aidldemo.model.Message; parcelable Message ;

切記 自定類型aidl文件名字、路徑須要和自定義類名字、路徑保持一致,

編譯項目,仍是報錯

parameter 1: 'Message message' can be an out parameter, so you must declare it as in, out or inout.

AIDL不是 Java。它是真的很接近,但它不是 Java。

Java 參數沒有方向的概念,AIDL 參數有方向,參數能夠從客戶端傳到服務端,再返回來。

若是sendMessage方法的message參數是純粹的輸入參數—這意味着是從客戶端到服務器的數據,你須要在AIDL聲明:

void sendMessage(in Message message);

若是sendMessage方法的message參數是純粹的輸出-這意味着它的數據是經過從服務器到客戶端,使用:

void sendMessage(out Message message);

若是sendMessage方法的message參數是輸入也是輸出-客戶端的值在服務可能會修改,使用:

void sendMessage(inout Message message);

總結

Android Studio Service AIDL 詳解 就到這裏,如今應該可使用AIDL實現想要的功能了,實現簡答的AIDL很簡單,主要是在自定義類型的時候,有幾個坑注意就好。

源碼地址: https://github.com/quanke/AIDLDemo.git

相關文章
相關標籤/搜索