日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Android AIDL實現與服務相互調用方式

瀏覽:29日期:2022-09-26 11:31:58

通過AIDL接口在進程間傳遞數據,記錄在開發中遇到的一寫問題

AIDL支持數據類型如下:

1. Java 的原生類型

2. String 和CharSequence

3. List 和 Map ,List和Map 對象的元素必須是AIDL支持的數據類型; 以上三種類型都不需要導入(import)

4. AIDL 自動生成的接口 需要導入(import)

5. 實現android.os.Parcelable 接口的類. 需要導入(import)。

問題1 在傳遞非基礎數據時 在參數前需加修飾符

void getDatas(in byte[] bs); void DataWhole(in PackageData data);}

這里重點是in、out、inout修飾符以及Parcelable的使用!常見的是in、Parcelable,少用的out、inout。

這幾種修飾符,可理解如下:

in:客戶端的參數輸入;

out:服務端的參數輸入;

inout:這個可以叫輸入輸出參數,客戶端可輸入、服務端也可輸入??蛻舳溯斎肓藚档椒斩撕?,服務端也可對該參數進行修改等,最后在客戶端上得到的是服務端輸出的參數。

問題2 傳遞對象時的必要操作

1.必需實現Parcelable接口,內部類必需為靜態內部類

2.需在aidl目錄創建同類名的AIDL文件,并聲明Parcelable,如圖

Android AIDL實現與服務相互調用方式

AIDL文件代碼就兩行

Android AIDL實現與服務相互調用方式

問題3 參數大小的限制

如上在傳遞byte[] 長度大于1024*1024時會拋出 TransactionTooLargeException 異常

問題4 實現與服務之間互相調用

1.在綁定服務時會返回一個實現了AIDL的對象,這樣可以通過對象調用服務中對應實現,

2.可以在應用層實現一個AIDL接口的對象,通過綁定服務返回的AIDL對象回傳給服務,這樣可以在服務中主動調用應用層的方法實現數據回傳通知,

//接收回調INotification notification = new INotification.Stub() { @Override public void Datas(byte[] bs) throws RemoteException { Log.d(TAG, 'Datas: 收到數據=' + Arrays.toString(bs));//已測試 最大數據1024*1024 } }

//傳遞回調對象void setNotification(in INotification Notification);

@Overridepublic void onServiceConnected(ComponentName name, IBinder service) { iAidlInterface = IAidlInterface.Stub.asInterface(service); try { iAidlInterface.setNotification(notification); } catch (RemoteException e) { e.printStackTrace(); }}

補充知識:在Android系統中實現AIDL功能

之前實現AIDL的功能都是通過eclipse或者android studio工具實現,最近由于項目需要,需要系統層提供接口給應用層使用,所以想到使用AIDL。下面已一個非常簡單的Demo來說明在Android系統平臺生成AIDL的jar供應用層使用。

一、AIDL的jar制作

首先新建一個android項目來用生產aidl的jar包,項目結構如下:

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主要是服務器端來實現的,而VoiceManager.java是供客戶端調用face方法使用的。VoiceManager.java具體實現如下:

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()){@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case MSG_INIT_SERVICE:removeMessages(MSG_INIT_SERVICE);break; default:break;}}};}//獲取服務端注冊的NAME服務并跟服務端建立連接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;} //調用服務端的face方法,實現兩個不同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 := simpleLOCAL_PACKAGE_NAME := SimpleServiceLOCAL_CERTIFICATE :=platforminclude $(BUILD_PACKAGE)

將該項目放置到android系統的packages/apps目錄單編就可以生產out/target/common/obj/JAVA_LIBRARIES/SimpleJar_intermediates/classes.jar,classes.jar就可以導入eclipse或者Android Studio中使用。

二、服務端實現AIDL中的接口demo目錄結構如下:

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

主要實現只有5個文件:SimpleService.java、Logger.java、BootReceiverBroadcast.java、 Android.mk、 AndroidManifest.xml。SimpleService.java是實現AIDL的服務,具體實現如下:

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;@Overridepublic IBinder onBind(Intent intent) {Logger.d();return mBinder;//跟客戶端綁定}@Overridepublic void onCreate() {super.onCreate();Logger.d();if (null == mBinder){initService();}}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {Logger.d();if (null == mBinder){initService();}return START_STICKY;}//實現AIDL的接口private class VoiceClientInterfaceImpl extends IVoiceClientInterface.Stub{@Overridepublic void face() throws RemoteException {Logger.d('face----excute!');//客戶端調用face方法時這里會執行,會打印face----excute!}}//初始化服務,主要是向系統注冊服務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服務,具體實現如下:

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);//開機啟動會拉起服務SimpleService context.startService(service); }}

Android.mk具體實現如下:

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS) LOCAL_SRC_FILES := $(call all-java-files-under, src)LOCAL_PACKAGE_NAME := SimpleServiceLOCAL_CERTIFICATE :=platformLOCAL_PRIVILEGED_MODULE := falseLOCAL_DEX_PREOPT := falseLOCAL_STATIC_JAVA_LIBRARIES := simpleinclude $(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>

到這里服務端就實現完了。

三、客戶端實現AIDL的接口調用demo目錄結構如下:

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跟服務端代碼一樣的。TestVoice.java的實現也很簡單,在button調用face方法,具體實現如下:

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實現如下:

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)LOCAL_SRC_FILES := $(call all-java-files-under, src)LOCAL_STATIC_JAVA_LIBRARIES := simpleLOCAL_PREBUILT_STATIC_JAVA_LIBRARIES := simple.jar #LOCAL_MODULE_TAGS :=optionalLOCAL_PACKAGE_NAME := Hello#LOCAL_CERTIFICATE :=platform#LOCAL_PRIVILEGED_MODULE := false#LOCAL_DEX_PREOPT := falseinclude $(BUILD_PACKAGE)

AndroidManifest.xml實現如下:

<?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>

到這里客戶端也實現了。將服務端跟客戶端的apk安裝到系統就可以測試了。

測試結果打印如下:

Android AIDL實現與服務相互調用方式

以上這篇Android AIDL實現與服務相互調用方式就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲影视一区| 久久精品国内一区二区三区水蜜桃| 黄色av日韩| 日本综合字幕| 99久久九九| 国产精品嫩草99av在线| 美女精品在线观看| 日韩三级一区| 免费在线成人| 精品国产精品久久一区免费式| 国内精品麻豆美女在线播放视频| 精品免费在线| 欧美影院三区| 久久亚洲美女| 欧美日韩亚洲国产精品| 国产精品a级| 欧美日韩国产v| 尤物在线精品| 亚洲乱码一区| 欧美激情日韩| 91精品国产乱码久久久久久久| 亚洲一区二区动漫| 日本免费一区二区视频| 久久精品人人| 国精品一区二区| 综合干狼人综合首页| 国产精品一区高清| 欧美香蕉视频| 石原莉奈在线亚洲二区| 91国内精品| 吉吉日韩欧美| 亚洲欧美网站| 麻豆国产一区| 中文在线免费视频| 亚洲一区二区三区四区五区午夜 | 国产精品传媒麻豆hd| 国精品产品一区| 国产精品毛片| 麻豆精品新av中文字幕| 免费观看不卡av| 久久国产尿小便嘘嘘| 免费在线小视频| 色综合视频一区二区三区日韩| 久久久免费人体| 亚洲精品午夜av福利久久蜜桃| 日本精品久久| 美女一区网站| 亚洲精品在线a| 97精品视频在线看| 亚洲精品高潮| 99精品视频在线观看免费播放| 日韩欧美中文在线观看| 日韩高清不卡| 欧美日韩午夜电影网| 啪啪国产精品| 国产精品一区二区三区美女| 午夜欧美视频| 国产成人调教视频在线观看| 一区二区三区四区日韩| 鲁鲁在线中文| 国产日韩视频在线| 国产高清久久| 国产精品久久亚洲不卡| 亚洲精品成人| 精品淫伦v久久水蜜桃| 午夜宅男久久久| 中文字幕人成乱码在线观看 | 欧美亚洲日本精品| 日韩动漫一区| 亚洲国产一区二区在线观看| 麻豆久久一区| 蜜臀久久久久久久| 中文字幕在线看片| 国产欧美一区二区三区精品观看| 五月婷婷亚洲| 精品不卡一区| 国产免费av一区二区三区| 国产精品日韩久久久| 国产不卡精品在线| 欧美伊人久久| 一区二区91| 不卡av一区二区| 欧美三级精品| 福利在线一区| 国产精品xvideos88| 天堂久久一区| 亚洲一区久久| 久久天堂成人| av免费不卡国产观看| 国产精品欧美大片| 婷婷久久免费视频| 国产精品嫩草99av在线| 国产亚洲一区二区手机在线观看| 国产精品高潮呻吟久久久久| 亚洲精品一级| 亚洲少妇一区| 黄色不卡一区| 国产精品av久久久久久麻豆网| 成人在线视频免费| 免费一级欧美片在线观看网站 | 国产农村妇女精品一二区| 日韩精品影视| 桃色一区二区| 精品欧美一区二区三区在线观看| 激情综合五月| 高清一区二区| 国产在线不卡一区二区三区| 久久爱www成人| 久久不卡日韩美女| 美女在线视频一区| 国产成人免费视频网站视频社区| 精品一区二区三区亚洲| 久久三级中文| 精品国产乱码| 国产+成+人+亚洲欧洲在线| 欧美激情综合| 精品久久久久久久| 国产一区二区三区亚洲综合| 你懂的网址国产 欧美| 狂野欧美性猛交xxxx| 久久精品一区| 国产精品久久久久久久免费观看| 福利片在线一区二区| 成人午夜亚洲| 成人免费电影网址| 亚洲二区免费| 中文久久精品| 日韩美女精品| 国产精品亚洲欧美一级在线| 欧美日本二区| 国产精区一区二区| 麻豆精品久久久| 91亚洲国产高清| 欧美 日韩 国产一区二区在线视频 | 日日夜夜免费精品视频| 综合激情婷婷| 国产精品一区二区三区av麻| 国产日产一区| 精品亚洲成人| 久久国产日韩| 免费的成人av| 亚洲a级精品| 久久99蜜桃| se01亚洲视频 | 超碰成人av| 欧美成人精品| 伊人久久一区| 欧美日韩一视频区二区| 久久精品二区亚洲w码 | 国产福利电影在线播放| 日韩在线欧美| 一本色道精品久久一区二区三区| 亚洲久久视频| 老色鬼精品视频在线观看播放| 精精国产xxxx视频在线播放| 黄色成人精品网站| 日本精品一区二区三区在线观看视频| 国产精品综合| 日韩在线免费| 免费精品视频最新在线| 91在线成人| 精品日韩在线| 国产精品普通话对白| 国产精品主播| 成人久久一区| 日韩精品社区| 日韩中文字幕高清在线观看| 亚洲一级淫片| 高清一区二区| 免费在线欧美视频| 欧美91在线| 国产精品日韩久久久| 国产精品videosex极品| 亚洲一级高清| 国产精品探花在线观看| 在线看片不卡| 国产黄色一区| 在线综合亚洲| 国精品产品一区| 亚洲开心激情| 老牛国内精品亚洲成av人片| 成人av二区| 免费看久久久| 丝袜亚洲精品中文字幕一区| 国产精品17p| 日韩天堂av| 久久一区亚洲| 综合精品一区| 神马日本精品| 国产精品色婷婷在线观看| 激情91久久| 精品视频在线一区二区在线| 亚洲一区不卡| 91av亚洲| 国产精品一区二区三区美女 | 亚洲人成在线影院| 日韩成人亚洲| 国产精品最新| 三级欧美在线一区| 日韩天堂在线|