跨進(jìn)程高并發(fā)時(shí)使用
流程
服務(wù)端:創(chuàng)建一個(gè)service來(lái)監(jiān)聽(tīng)客戶端的鏈接請(qǐng)求,然后創(chuàng)建一個(gè)AIDL文件个榕,暴露給客戶端赖捌,把想要暴露給客戶端的接口在這里聲明扛拨,最后在service中實(shí)現(xiàn)這個(gè)AIDL接口
客戶端:綁定服務(wù)端service猫缭,將服務(wù)端返回的binder對(duì)象轉(zhuǎn)成AIDL接口所屬類型葱弟,接著就可以調(diào)用AIDL中的方法了
AIDL支持的數(shù)據(jù)類型
1.基本數(shù)據(jù)類型
2.String和CharSequence
3.List:只支持ArrayList,每個(gè)元素必須被AIDL支持
4.Map:只支持hashMap,每個(gè)元素必須被AIDL支持(key&vaule)
5.Parcelable:所有實(shí)現(xiàn)Parecelable接口的對(duì)象
6.AIDL:所有AIDL接口本身也可以在AIDL文件中使用
語(yǔ)法
1.所有使用到的對(duì)象必須import進(jìn)來(lái)猜丹,不管是不是在同一個(gè)包下
2.如果AIDL文件使用到了自定義的Parcelable對(duì)象芝加,那么必須新建一個(gè)和它同名的AIDl文件,并且聲明為Parcelable類型
例如:BookManager.aidl 和 Book.aidl
package com.smt.demo;
parcelable Book;
3.不支持靜態(tài)常量
4.方法參數(shù)必須加上 in,out,inout
in:輸入類型參數(shù)(在service中改變?cè)搶?duì)象屬性射窒,原始對(duì)象不會(huì)改變)
out:輸出類型參數(shù)(在service中改變?cè)搶?duì)象屬性藏杖,原始對(duì)象會(huì)改變)
inout:輸入輸出類型參數(shù)
根據(jù)情況使用,out&inout是有開(kāi)銷的
5.AIDL包接口需要一致(客戶端&服務(wù)端)脉顿,客戶端中需要反序列化服務(wù)端中的AIDL接口相關(guān)的類蝌麸,可以把AIDL文件放到同一個(gè)包下,方便移植
使用as創(chuàng)建AIDL文件艾疟,找不到類問(wèn)題
解答:在build.gradle 文件 Android 中加入
sourceSets {
main {
java.srcDirs = ['src/main/java', 'src/main/aidl']
}
}
具體語(yǔ)法請(qǐng)參考BookManager.aidl Book.aidl
注意事項(xiàng)
1.在實(shí)現(xiàn)services中方法時(shí)祥楣,需要加同步,因?yàn)锳IDL方法是在服務(wù)端的Binder線程池中執(zhí)行的汉柒,當(dāng)多個(gè)客戶端同時(shí)連接時(shí)會(huì)出現(xiàn)并發(fā)訪問(wèn)
BookManager.Stub mBookManager = new BookManager.Stub() {
@Override
public List<Book> getBooks() throws RemoteException {
synchronized (mBooks){
return mBooks;
}
}
}误褪;
2.對(duì)象不能跨進(jìn)程傳輸,本質(zhì)上是對(duì)象的反序列化過(guò)程(A 對(duì)象傳到其他進(jìn)程時(shí)碾褂,是重新創(chuàng)建的一個(gè)A對(duì)象兽间,兩個(gè)A對(duì)象不相等)
觀察者模式解綁遇到的問(wèn)題
可以使用RemoteCallbackList解決,RemoteCallbackList<E extends IInterface>里面維護(hù)了一個(gè)
ArrayMap<IBinder, Callback>,key為IBinder正塌,value為Callback嘀略,雖然多次跨進(jìn)程傳輸客戶端
的同一個(gè)對(duì)象會(huì)在服務(wù)端生成不同的對(duì)象,但是這些新對(duì)象的Binder對(duì)象是同一個(gè)乓诽,當(dāng)時(shí)客戶端
進(jìn)程掛掉之后會(huì)自動(dòng)幫我們移出Listenr帜羊,RemoteCallbackList內(nèi)部實(shí)現(xiàn)了現(xiàn)場(chǎng)同步的功能
使用方式:
RemoteCallbackList<BookListener> callbackList = new RemoteCallbackList();
int count = callbackList.beginBroadcast();
for (int i = 0; i < count ; i++) {
BookListener broadcastItem = callbackList.getBroadcastItem(i);
//TODO you code...
}
callbackList.finishBroadcast();
3.盡量不要在UI線程調(diào)用服務(wù)端的遠(yuǎn)程方法(可能會(huì)導(dǎo)致ANR),onServiceConnected和onServiceDisconnected方法也都運(yùn)行在UI線程中鸠天,避免調(diào)用服務(wù)端耗時(shí)方法
4.由于服務(wù)端方法是運(yùn)行的binder線程池中的讼育,所以可以執(zhí)行耗時(shí)方法,這時(shí)輕易不要開(kāi)現(xiàn)場(chǎng)去執(zhí)行稠集,當(dāng)服務(wù)端調(diào)用客戶端的方法時(shí)奶段,客戶端方法運(yùn)行在客戶端的binder線程池中,所以不能做耗時(shí)操作和更新UI操作
5.當(dāng)Binder意外死亡時(shí)剥纷,可以選擇重連
兩種方式
1.DeathRecipient痹籍,當(dāng)Bander死亡時(shí)會(huì)回調(diào)binderDied,可在此方法進(jìn)行重連操
作
2.onServiceDisconnected中重連操作
兩種方式的區(qū)別:onServiceDisconnected運(yùn)行在UI線程晦鞋,DeathRecipient方式
運(yùn)行在客戶端的Binder線程中
6.權(quán)限驗(yàn)證:在onTransact方法中驗(yàn)證蹲缠,可以使用自定義權(quán)限棺克,包名uid,pic等來(lái)驗(yàn)證线定,返回false表示調(diào)用服務(wù)端方法失斈纫辍(需要拷貝AIDL生成的java文件進(jìn)行修改)