Android AIDL實現(xiàn)與服務(wù)相互調(diào)用方式
通過AIDL接口在進程間傳遞數(shù)據(jù),記錄在開發(fā)中遇到的一寫問題
AIDL支持數(shù)據(jù)類型如下:
1. Java 的原生類型
2. String 和CharSequence
3. List 和 Map ,List和Map 對象的元素必須是AIDL支持的數(shù)據(jù)類型; 以上三種類型都不需要導入(import)
4. AIDL 自動生成的接口 需要導入(import)
5. 實現(xiàn)android.os.Parcelable 接口的類. 需要導入(import)。
問題1 在傳遞非基礎(chǔ)數(shù)據(jù)時 在參數(shù)前需加修飾符
void getDatas(in byte[] bs); void DataWhole(in PackageData data); }
這里重點是in、out、inout修飾符以及Parcelable的使用!常見的是in、Parcelable,少用的out、inout。
這幾種修飾符,可理解如下:
in:客戶端的參數(shù)輸入;
out:服務(wù)端的參數(shù)輸入;
inout:這個可以叫輸入輸出參數(shù),客戶端可輸入、服務(wù)端也可輸入??蛻舳溯斎肓藚?shù)到服務(wù)端后,服務(wù)端也可對該參數(shù)進行修改等,最后在客戶端上得到的是服務(wù)端輸出的參數(shù)。
問題2 傳遞對象時的必要操作
1.必需實現(xiàn)Parcelable接口,內(nèi)部類必需為靜態(tài)內(nèi)部類
2.需在aidl目錄創(chuàng)建同類名的AIDL文件,并聲明Parcelable,如圖

AIDL文件代碼就兩行

問題3 參數(shù)大小的限制
如上在傳遞byte[] 長度大于1024*1024時會拋出 TransactionTooLargeException 異常
問題4 實現(xiàn)與服務(wù)之間互相調(diào)用
1.在綁定服務(wù)時會返回一個實現(xiàn)了AIDL的對象,這樣可以通過對象調(diào)用服務(wù)中對應(yīng)實現(xiàn),
2.可以在應(yīng)用層實現(xiàn)一個AIDL接口的對象,通過綁定服務(wù)返回的AIDL對象回傳給服務(wù),這樣可以在服務(wù)中主動調(diào)用應(yīng)用層的方法實現(xiàn)數(shù)據(jù)回傳通知,
//接收回調(diào)
INotification notification = new INotification.Stub() {
@Override
public void Datas(byte[] bs) throws RemoteException {
Log.d(TAG, "Datas: 收到數(shù)據(jù)=" + Arrays.toString(bs));//已測試 最大數(shù)據(jù)1024*1024
}
}
//傳遞回調(diào)對象 void setNotification(in INotification Notification);
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
iAidlInterface = IAidlInterface.Stub.asInterface(service);
try {
iAidlInterface.setNotification(notification);
} catch (RemoteException e) {
e.printStackTrace();
}
}
補充知識:在Android系統(tǒng)中實現(xiàn)AIDL功能
之前實現(xiàn)AIDL的功能都是通過eclipse或者android studio工具實現(xiàn),最近由于項目需要,需要系統(tǒng)層提供接口給應(yīng)用層使用,所以想到使用AIDL。下面已一個非常簡單的Demo來說明在Android系統(tǒng)平臺生成AIDL的jar供應(yīng)用層使用。
一、AIDL的jar制作
首先新建一個android項目來用生產(chǎn)aidl的jar包,項目結(jié)構(gòu)如下:
gunder@gunder:/mnt/hgfs/ubuntuShare/aidl/SimpleJar$ tree . ├── Android.mk └── src └── com └── china └── jar ├── IVoiceClientInterface.aidl └── VoiceManager.java
只有三個文件,首先看一下IVoiceClientInterface.aidl文件:
package com.china.jar;
interface IVoiceClientInterface{
void face();
}
里面只有一個簡單的方法face。 IVoiceClientInterface.aidl主要是服務(wù)器端來實現(xiàn)的,而VoiceManager.java是供客戶端調(diào)用face方法使用的。VoiceManager.java具體實現(xiàn)如下:
package com.china.jar;
import com.china.jar.IVoiceClientInterface;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Message;
import android.os.RemoteException;
import android.util.Log;
import android.os.ServiceManager;
public class VoiceManager {
private static final String TAG = "VoiceManager";
private static VoiceManager mVoiceManager;
private static IVoiceClientInterface mService = null;
public static final String NAME = "simple_jar";
public static final boolean DEBUG_DATA = true;
private final HandlerThread mWorkThread;
private final Handler mWorkHander;
private static final int MSG_INIT_SERVICE = 0x01;
//單例模式
public static synchronized VoiceManager getInstance(){
if (null == mVoiceManager){
synchronized (VoiceManager.class) {
if (null == mVoiceManager){
mVoiceManager = new VoiceManager();
}
}
}
return mVoiceManager;
}
private VoiceManager(){
mWorkThread = new HandlerThread("simple_manager");
mWorkThread.start();
mWorkHander = new Handler(mWorkThread.getLooper()){
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case MSG_INIT_SERVICE:
removeMessages(MSG_INIT_SERVICE);
break;
default:
break;
}
}
};
}
//獲取服務(wù)端注冊的NAME服務(wù)并跟服務(wù)端建立連接
private synchronized IVoiceClientInterface getService(){
if (null == mService){
Log.e(TAG, "IVocieService init");
mService = IVoiceClientInterface.Stub.asInterface(ServiceManager
.getService(NAME));
}
if (null == mService){
Log.e(TAG, "jar service is null");
mWorkHander.removeMessages(MSG_INIT_SERVICE);
mWorkHander.sendEmptyMessageDelayed(MSG_INIT_SERVICE, 100);
}
return mService;
}
//調(diào)用服務(wù)端的face方法,實現(xiàn)兩個不同app之間的進程間通信
public void face(){
Log.d(TAG, "face");
mService = getService();
if (null == mService){
Log.e(TAG, "face mService is null!");
return ;
}
try{
mService.face();
}catch(RemoteException e){
e.printStackTrace();
}
}
}
Android.mk文件主要是用來將IVoiceClientInterface.aidl和VoiceManager.java編譯成jar包,以方便在eclipse或者Android Studio中使用。
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_SRC_FILES := $(call all-java-files-under, src) LOCAL_JAVA_LIBRARIES := simple LOCAL_PACKAGE_NAME := SimpleService LOCAL_CERTIFICATE :=platform include $(BUILD_PACKAGE)
將該項目放置到android系統(tǒng)的packages/apps目錄單編就可以生產(chǎn)out/target/common/obj/JAVA_LIBRARIES/SimpleJar_intermediates/classes.jar,classes.jar就可以導入eclipse或者Android Studio中使用。
二、服務(wù)端實現(xiàn)AIDL中的接口demo目錄結(jié)構(gòu)如下:
gunder@gunder:/mnt/hgfs/ubuntuShare/aidl/SimpleJarService$ tree
.
├── AndroidManifest.xml
├── Android.mk
├── libs
│ └── simple.jar
├── res
│ ├── drawable-hdpi
│ │ └── ic_launcher.png
│ ├── drawable-ldpi
│ ├── drawable-mdpi
│ │ └── ic_launcher.png
│ ├── drawable-xhdpi
│ │ └── ic_launcher.png
│ ├── layout
│ ├── values
│ │ ├── strings.xml
│ │ └── styles.xml
│ ├── values-v11
│ │ └── styles.xml
│ └── values-v14
│ └── styles.xml
└── src
└── com
└── china
└── service
├── BootReceiverBroadcast.java
├── Logger.java
└── SimpleService.java
主要實現(xiàn)只有5個文件:SimpleService.java、Logger.java、BootReceiverBroadcast.java、 Android.mk、 AndroidManifest.xml。SimpleService.java是實現(xiàn)AIDL的服務(wù),具體實現(xiàn)如下:
package com.china.service;
import com.china.jar.IVoiceClientInterface;
import com.china.jar.VoiceManager;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
import android.os.ServiceManager;
public class SimpleService extends Service{
private static VoiceClientInterfaceImpl mBinder;
@Override
public IBinder onBind(Intent intent) {
Logger.d();
return mBinder;//跟客戶端綁定
}
@Override
public void onCreate() {
super.onCreate();
Logger.d();
if (null == mBinder){
initService();
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Logger.d();
if (null == mBinder){
initService();
}
return START_STICKY;
}
//實現(xiàn)AIDL的接口
private class VoiceClientInterfaceImpl extends IVoiceClientInterface.Stub{
@Override
public void face() throws RemoteException {
Logger.d("face----excute!");//客戶端調(diào)用face方法時這里會執(zhí)行,會打印face----excute!
}
}
//初始化服務(wù),主要是向系統(tǒng)注冊服務(wù)
private void initService(){
Logger.d();
if (null == mBinder){
synchronized (SimpleService.class) {
if (null == mBinder){
try {
mBinder = new VoiceClientInterfaceImpl();
ServiceManager.addService(VoiceManager.NAME, mBinder);
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
}
Logger.java是打印Log的簡單封裝,具體如下:
package com.china.service;
import android.util.Log;
import java.util.Locale;
public class Logger {
public static final boolean DEBUG = true;
public static final String DEFAULT_TAG = "AIOS_";
public Logger(){}
public static void d(){
if (DEBUG){
Log.d(DEFAULT_TAG,getPrefix());
}
}
public static void d(String msg){
if (DEBUG){
Log.d(DEFAULT_TAG, getPrefix() + msg);
}
}
public static void d(String msg, Throwable tr){
if (DEBUG){
Log.d(DEFAULT_TAG, getPrefix() + msg, tr);
}
}
private static String getPrefix(){
StackTraceElement stackTraceElement = Thread.currentThread().getStackTrace()[4];
String className = stackTraceElement.getClassName();
int classNameStartIndex = className.lastIndexOf(".") + 1;
className = className.substring(classNameStartIndex);
String methodName = stackTraceElement.getMethodName();
int methodLine = stackTraceElement.getLineNumber();
String format = "%s_%s(L:%d)";
return String.format(Locale.CANADA, format, className, methodName, methodLine);
}
}
BootReceiverBroadcast.java是開機完成的時候拉起 SimpleService服務(wù),具體實現(xiàn)如下:
package com.china.service;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
public class BootReceiverBroadcast extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
Logger.d();
Intent service = new Intent(context, SimpleService.class);//開機啟動會拉起服務(wù)SimpleService
context.startService(service);
}
}
Android.mk具體實現(xiàn)如下:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_SRC_FILES := $(call all-java-files-under, src) LOCAL_PACKAGE_NAME := SimpleService LOCAL_CERTIFICATE :=platform LOCAL_PRIVILEGED_MODULE := false LOCAL_DEX_PREOPT := false LOCAL_STATIC_JAVA_LIBRARIES := simple include $(BUILD_PACKAGE) include $(CLEAR_VARS) LOCAL_MODULE_TAGS := optional LOCAL_PREBUILT_STATIC_JAVA_LIBRARIES :=simple:libs/simple.jar include $(BUILD_MULTI_PREBUILT) include $(call all-makefiles-under,$(LOCAL_PATH))
這里的simple.jar是第一步中制作的classes.jar。 AndroidManifest.xml配置文件如下:
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.chinatsp.service" android:versionCode="1" android:versionName="1.0" android:sharedUserId="android.uid.system" > <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="21" /> <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" > <service android:name="com.china.service.SimpleService"></service> <receiver android:name="com.china.service.BootReceiverBroadcast"> <intent-filter> <action android:name="android.intent.action.BOOT_COMPLETED"/> <!-- <category android:name="android.intent.category.LAUNCHER"/> --> </intent-filter> </receiver> </application> </manifest>
到這里服務(wù)端就實現(xiàn)完了。
三、客戶端實現(xiàn)AIDL的接口調(diào)用demo目錄結(jié)構(gòu)如下:
gunder@gunder:/mnt/hgfs/ubuntuShare/aidl/SimpleJarClient$ tree
.
├── AndroidManifest.xml
├── Android.mk
├── libs
│ └── simple.jar
├── res
│ ├── drawable-hdpi
│ │ └── ic_launcher.png
│ ├── drawable-ldpi
│ ├── drawable-mdpi
│ │ └── ic_launcher.png
│ ├── drawable-xhdpi
│ │ └── ic_launcher.png
│ ├── drawable-xxhdpi
│ │ └── ic_launcher.png
│ ├── layout
│ │ ├── activity_main.xml
│ │ ├── activity_tss.xml
│ │ └── test.xml
│ ├── menu
│ ├── values
│ │ ├── dimens.xml
│ │ └── strings.xml
│ ├── values-v11
│ ├── values-v14
│ └── values-w820dp
│ └── dimens.xml
└── src
└── com
└── example
└── helloworld
├── TestVoice.java
└── util
└── Logger.java
這里主要看5個文件:Logger.java、 test.xml、TestVoice.java、Android.mk、AndroidManifest.xml,其中Logger.java跟服務(wù)端代碼一樣的。TestVoice.java的實現(xiàn)也很簡單,在button調(diào)用face方法,具體實現(xiàn)如下:
package com.example.helloworld;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import com.example.helloworld.util.Logger;
public class TestVoice extends Activity{
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.test);
}
public void startVoice(View view){
Logger.d();
}
public void stopVoice(View view){
Logger.d();
com.china.jar.VoiceManager.getInstance().face();
}
public void finishVoice(View view){
Logger.d();
finish();
}
}
test.xml布局如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="startVoice" android:text="@string/tts_start"/> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="stopVoice" android:text="@string/tts_stop"/> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="finishVoice" android:text="@string/tts_finish"/> </LinearLayout>
Android.mk實現(xiàn)如下:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_SRC_FILES := $(call all-java-files-under, src) LOCAL_STATIC_JAVA_LIBRARIES := simple LOCAL_PREBUILT_STATIC_JAVA_LIBRARIES := simple.jar #LOCAL_MODULE_TAGS :=optional LOCAL_PACKAGE_NAME := Hello #LOCAL_CERTIFICATE :=platform #LOCAL_PRIVILEGED_MODULE := false #LOCAL_DEX_PREOPT := false include $(BUILD_PACKAGE)
AndroidManifest.xml實現(xiàn)如下:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.helloworld" android:versionCode="1" android:versionName="1.0" > <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> <uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.BLUETOOTH" /> <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" /> <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" /> <uses-permission android:name="android.permission.WRITE_SETTINGS"/> <uses-sdk android:minSdkVersion="8" android:targetSdkVersion="21" /> <application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" > <activity android:name="com.example.helloworld.TestVoice" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest>
到這里客戶端也實現(xiàn)了。將服務(wù)端跟客戶端的apk安裝到系統(tǒng)就可以測試了。
測試結(jié)果打印如下:

以上這篇Android AIDL實現(xiàn)與服務(wù)相互調(diào)用方式就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
詳解SwipeListView框架實現(xiàn)微信\QQ滑動刪除效果
這篇文章主要為大家詳細介紹了SwipeListView框架實現(xiàn)微信\QQ滑動刪除效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-08-08
Android搜索結(jié)果顯示高亮實例(有數(shù)據(jù)滑動底部自動刷新)
本篇文章主要介紹了Android搜索結(jié)果顯示高亮實例(有數(shù)據(jù)滑動底部自動刷新),非常具有實用價值,需要的朋友可以參考下2017-04-04
Android+SQLite數(shù)據(jù)庫實現(xiàn)的生詞記事本功能實例
這篇文章主要介紹了Android+SQLite數(shù)據(jù)庫實現(xiàn)的生詞記事本功能,結(jié)合具體實例形式分析了Android操作SQLite數(shù)據(jù)庫實現(xiàn)生詞記錄功能的操作步驟與相關(guān)注意事項,需要的朋友可以參考下2017-09-09
Android混合開發(fā)教程之WebView的使用方法總結(jié)
WebView是一個基于webkit引擎、展現(xiàn)web頁面的控件,下面這篇文章主要給大家介紹了關(guān)于Android混合開發(fā)教程之WebView的使用方法,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考借鑒,下面來一起看看吧2018-05-05

