簡(jiǎn)介
AIDL(Android Interface Definition Language)使客戶端和服務(wù)端通過它定義的編程接口來達(dá)成共識(shí),以便進(jìn)行進(jìn)程間通信(IPC)。
僅當(dāng)允許其它應(yīng)用程序通過IPC方式訪問Service,并且Service需要多線程運(yùn)行時(shí)旦事,才應(yīng)該使用AIDL试浙。
- 如果從本地進(jìn)程發(fā)起調(diào)用,則調(diào)用將在調(diào)用線程中運(yùn)行捏境。如果想要本地調(diào)用依然在另一個(gè)進(jìn)程中運(yùn)行,可以在清單文件的Service中聲明android:process=":remote"毁葱。
- 如果從遠(yuǎn)程進(jìn)程發(fā)起調(diào)用垫言,則會(huì)在Service的進(jìn)程運(yùn)行。
- 關(guān)鍵字oneway會(huì)改變遠(yuǎn)程調(diào)用的處理方式(該關(guān)鍵字只能作用于某一個(gè)接口)倾剿。遠(yuǎn)程調(diào)用不會(huì)阻塞筷频,但本地調(diào)用仍然是同步執(zhí)行的蚌成。
使用
1. 創(chuàng)建.aidl文件
2. 服務(wù)端實(shí)現(xiàn)接口并向客戶端公布
3. 客戶端獲取接口實(shí)例
4. 服務(wù)端調(diào)用客戶端
5. Parcelable
<a href="#創(chuàng)建.aidl文件">1. 創(chuàng)建.aidl文件</a>
在.aidl文件中,聲明一個(gè)帶有若干方法的接口凛捏。且只能定義一個(gè)接口担忧,且只能包含接口的定義和方法聲明。
聲明的方法可帶有參數(shù)和返回值坯癣,它們的類型可以為下面允許的類型瓶盛。
所有非基本類型的參數(shù)都需要帶有一個(gè)指明數(shù)據(jù)方向的標(biāo)志。
可以是in示罗、out惩猫、inout。
簡(jiǎn)單類型的參數(shù)默認(rèn)是in鹉勒,且不能是其他方向的值帆锋。
<a href="#允許的類型">允許的類型</a>
- Java語(yǔ)言的所有基本類型,void禽额,null
- CharSequence
- String
- List,類型實(shí)參必須是這里列出的類型锯厢。實(shí)際使用的類是ArrayList
- Map,類型實(shí)參必須是這里列出的類型脯倒。實(shí)際使用的類是HashMap
- 基于AIDL生成的接口
- 已聲明的Parcelable類型
最后把.aidl文件放到src/目錄下实辑,在編譯時(shí),SDK工具會(huì)在gen/目錄下生成IBinder接口文件藻丢。
<a href="#服務(wù)端實(shí)現(xiàn)接口并向客戶端公布">2. 服務(wù)端實(shí)現(xiàn)接口并向客戶端公布</a>
.aidl文件被自動(dòng)生成.java接口文件剪撬。它包含一個(gè)靜態(tài)子類Stub,該子類擴(kuò)展了Binder類悠反,且實(shí)現(xiàn)父接口残黑。
所以,我們應(yīng)該擴(kuò)展Stub斋否,并實(shí)現(xiàn)由.aidl文件繼承的方法梨水。
在實(shí)現(xiàn)AIDL接口時(shí)的注意事項(xiàng):
- 此接口的實(shí)現(xiàn)必須要保證多線程運(yùn)行的安全
- 默認(rèn)情況下,客戶端調(diào)用該接口是以同步方式運(yùn)行的茵臭。所以該服務(wù)的響應(yīng)時(shí)間較長(zhǎng)疫诽,則不應(yīng)該在主線程中調(diào)用,而是在客戶端的單獨(dú)線程中調(diào)用旦委。
- 所有異常都不會(huì)發(fā)還給調(diào)用者
向客戶端公布接口
在onBind方法中返回該接口的實(shí)例奇徒。
<a href="#客戶端獲取接口實(shí)例">3. 客戶端獲取接口實(shí)例</a>
為了讓客戶端有權(quán)限訪問接口類,客戶端必須在src/目錄下?lián)碛幸环?aidl文件的拷貝缨硝,用于生成供客戶端訪問AIDL方法的Binder接口摩钙。
在接收到回調(diào)方法onServiceConnected中的IBinder時(shí),客戶端必須調(diào)用YourServiceInterface.Stub.asInterface(service)來把返回的參數(shù)轉(zhuǎn)換為YourServiceInterface類型查辩。
之后就可以使用該接口和服務(wù)端通信了腺律。
在調(diào)用服務(wù)器的方法時(shí)奕短,必須要catch DeadObjectException異常宜肉,當(dāng)連接中斷時(shí)會(huì)拋出該異常匀钧,這是遠(yuǎn)程方法唯一拋出的異常。
<a href=#服務(wù)端調(diào)用客戶端>4. 服務(wù)端調(diào)用客戶端</a>
通過之前的步驟谬返,客戶端已經(jīng)可以調(diào)用服務(wù)端的方法了之斯。如果服務(wù)端也想回調(diào)客戶端,需要更多的步驟遣铝。
- 除了之前建立的.aidl文件佑刷,還需要額外簡(jiǎn)歷一個(gè).aidl文件,該文件表示客戶端提供給服務(wù)端調(diào)用的接口酿炸。
- 之前的.aidl文件中需要增加兩個(gè)方法瘫絮,注冊(cè)和解注冊(cè)。參數(shù)為第二個(gè).aidl文件的接口填硕。
- 在服務(wù)端實(shí)現(xiàn)第一個(gè)接口麦萤。
- 實(shí)例化一個(gè)RemoteCallbackList,它是用來存儲(chǔ)客戶端接口的列表扁眯。
- 實(shí)現(xiàn)注冊(cè)和解注冊(cè)方法壮莹,一般為,判斷參數(shù)是否為null姻檀,不為null命满,則調(diào)用RemoteCallbackList的方法注冊(cè)/解注冊(cè)。
- 在客戶端實(shí)現(xiàn)第二個(gè)接口绣版。
<a href="#Parcelable">5. Parcelable</a>
如果想要在AIDL中使用某個(gè)類作為參數(shù)或返回值胶台,那么該類必須實(shí)現(xiàn)Parcelable接口,并且使用parcelable package.class;來聲明杂抽。
<a href="#實(shí)現(xiàn)Parcelable接口">實(shí)現(xiàn)Parcelable接口</a>
- 實(shí)現(xiàn)describeContents方法诈唬,默認(rèn)返回0就可以。
- 實(shí)現(xiàn)writeToParcel(Parcel dest, int flags)方法默怨。該方法用于序列化讯榕,即把類中的數(shù)據(jù)寫入外部的Parcel中保存。
- 實(shí)現(xiàn)靜態(tài)的Parcelable.Creator接口:
- createFormParcel(Parcel in)用于反序列化匙睹,該方法從Parcel中讀取數(shù)據(jù)以創(chuàng)建對(duì)象愚屁。
- newArray(int size)用于創(chuàng)建該類的一個(gè)數(shù)組,使用return new T[size]即可痕檬。
Parcelable和Serializable的區(qū)別
Parcelable直接在內(nèi)存中讀寫霎槐,而Serializable讀寫到硬盤上。
因此梦谜,Parcelable速度也更快丘跌。而Serializable使用了反射袭景,速度更慢。
但是因?yàn)镻arcelable只在內(nèi)存中讀寫闭树,所以耸棒,它并不能通過網(wǎng)絡(luò)傳遞,只能跨進(jìn)程使用报辱。而Serializable都可以与殃。
參考