Binder實(shí)現(xiàn)了IBinder接口,可理解為port一樣的虛擬設(shè)備滔灶,其驅(qū)動為/dev/binder
Binder是ServiceManager連接各種Manager(ActivityManager普碎、WindowManager等)和相應(yīng)ManagerService的橋梁
客戶端和服務(wù)器之間的媒介,當(dāng)bindservice時录平,服務(wù)器返回一個版喊了服務(wù)器業(yè)務(wù)調(diào)用的Binder對象麻车,由此對象可以獲取服務(wù)端提供給的服務(wù)或數(shù)據(jù)
首先
準(zhǔn)備生成AILD文件必要的1個.java文件和2個.aidl文件
1、實(shí)現(xiàn)了Parcelable接口的文件 .java
package com.example.aidltest
import android.os.Parcel
import android.os.Parcelable
public class Book implements Parcelable{
private int BookId;
private int BookName;
private boolean hasFinish;
public Book(int bookid斗这,int bookname动猬,boolean hasfinish){
this.BookId=bookid;
this.BookName=bookname;
this.hasFinish=hasfinish;
}
@override
public int describeContents(){
return 0; //基本都返回0
}
@override
public void writeToParcel(Parcel dest,int flags){
//對象寫入Parcel
dest.writeInt(BookId);
dest.writeString(BookName);
dest.writeInt(hasFinish?1:0);
}
public static final Create<Book> CREATE=new Creato<Book>(){
//此句需要自己寫 但是API中有
@override
public Book createFromParcel(Parcel in){
return new Book(in);
}
@override
public Book[] new Array(int size){
return Book[size];
}
};
protected Book(Parcel in){
//從Parcel(序列化后的對象)中創(chuàng)建原始對象
BookId=in.readInt();
BookName=in.readString();
hasFinish=in.readInt==1;
//要保證此處的引用的順序及類型和牽絆writeToParcel的引用和類型相對應(yīng)(特別要注意次序)
}
}
2表箭、與實(shí)現(xiàn)了Parcelable接口的類同名的aidl文件 .aidl
package com.example.aidltest
Parcelable Book;
3赁咙、包含想要在服務(wù)器(或其他進(jìn)程)提供數(shù)據(jù)或可供客戶端使用的方法的IBookManager.aidl文件 .aidl
package com.example.aidltest
import com.example.aidltest.Book
//要想使用實(shí)現(xiàn)了Parcelable接口的類則必須在此引入
interface IBookManager{ //定義一個接口
//以下為可以提供給客戶端調(diào)用的方法
List<Book> getBookList();
void addBook(in Book book); //除了八大基本數(shù)據(jù)類型,其他的對象都要使用in免钻、out彼水、inout來標(biāo)識
}
只能聲明方法,不能生命靜態(tài)常量
其次
make project或rebuild project
會自動生成對應(yīng)可使用的IBookManager.java接口文件
**此接口文件包含一個Stub內(nèi)部類和IBanderManger.aidl中聲明的方法
之后創(chuàng)建對象极舔、綁定binder凤覆、調(diào)用方法都通過此.java文件
1、聲明了定義在IBookManager.aidl中的方法拆魏,并同時聲明了整型id用來標(biāo)識這兩個方法
2盯桦、內(nèi)部類Stub。此類即為Binder類渤刃。
- 當(dāng)client與sevice同進(jìn)程時拥峦,方法調(diào)用不會走跨進(jìn)成的transact過程;
- 當(dāng)兩者不同進(jìn)程時溪掀,方法調(diào)用需走此邏輯事镣,由Stub的內(nèi)部代理類Proxy來完成
3步鉴、DESCRIPTOR
Binder唯一標(biāo)識符揪胃。一般用當(dāng)前Binder的類名表示
4璃哟、asInterface
用于將服務(wù)端的Binder對象轉(zhuǎn)換為客戶端所需的AIDL接口類對象;
同一進(jìn)程返回styb本身喊递,否則返回系統(tǒng)封裝后的Stub.Proxy對象
5随闪、asBinder
用于返回當(dāng)前Binder對象
6、onTransact
方法運(yùn)行在服務(wù)端Binder線程池中骚勘;客戶端跨進(jìn)成請求由系統(tǒng)底層封裝后交由此方法處理铐伴;
作用: - code可明確目標(biāo)
- data中提取出目標(biāo)方法所用參數(shù)(如果有)
- 執(zhí)行目標(biāo)方法
- 執(zhí)行完成后將返回值寫入reply(如果有),若返回false則請求失敗俏讹,可用來做權(quán)限認(rèn)證
7当宴、Proxy#getBookList、Proxy#addBook(無返回值)
運(yùn)行于客戶端 - 創(chuàng)建輸入型Parcel對象_data泽疆;輸出型Parcel對象_reply户矢;返回值List對象
- 將調(diào)用者的細(xì)數(shù)寫入_data(如果有)
- 調(diào)用transact方法發(fā)起RPC(遠(yuǎn)程過程調(diào)用),同時當(dāng)前線程掛起
- 服務(wù)端onTransact執(zhí)行殉疼,直至RPC過程返回后梯浪,當(dāng)前線程繼續(xù)進(jìn)行,從_reply中取出返回值
- 返回_reply中數(shù)據(jù)
注意
1瓢娜、當(dāng)client向service發(fā)起請求后當(dāng)前線程會掛起挂洛,所以如果很耗時就不要寫在UI線程中
2、service的Binder方法運(yùn)行在Binder線程池中眠砾,所以不論Binder方法是否耗時都要進(jìn)行同步操作
過程:
- Clint發(fā)起RPC后掛起
- Binder寫入?yún)?shù)(data)后調(diào)用transact
- Service調(diào)用onTransact并運(yùn)行在Binder線程池中
- 執(zhí)行完畢后返回結(jié)果寫入reply
- reply返回給Binder后喚醒client發(fā)起RPC的線程
Binder機(jī)制完全不需要AIDL文件虏劲,他們都只是為了方便自動生成出IBookManager.java而創(chuàng)建的
AIDL機(jī)制的使用
所需:
service:用來監(jiān)聽client的連接請求;然后創(chuàng)建一個AIDL實(shí)例褒颈;將暴露給client的接口于文件中聲明伙单,之后在Service中實(shí)現(xiàn)此接口即可
client:綁定Service;將service返回的Binder對象轉(zhuǎn)成AIDL接口所屬類型之后就可以調(diào)用方法
AIDL接口文件哈肖,前面創(chuàng)建的IBookManager.java
Service
public class BookManagerService extends Service{
private CopyOnWriteArrayList<Book> mBooklist=new CopyOnWriteArrayList<>();
//IBookManager.java的內(nèi)部類
private Binder mBinder = new IBookManager.Stub(){
@override
public List<Book> getBookList() throws RemoteException{
return mBooklist;
}
@override
public void addBook(Book book) throws RemoteException{
mBooklist.add(book);
}
};
@override
public void onCreate(){
super.onCreate();
mBooklist.add(new Book(1吻育,"ANDROID"));
mBooklist.add(new Book(2,"IOS"));
//演示用的淤井,在初始化時向mBooklist中添加兩個元素
}
@0verride
public IBinder onBind(Intent intent){
return mBinder;
//前面的IBookManager內(nèi)部類Stub的對象(此對象為Binder對象)
}
}
Client
public class BookManagerActivity extends AppCompatActivity{
private ServiceConnection mConnection=new ServiceConnection(){
public void onServiceConnected(ComponentName className布疼,IBinder **service**){
IBookManager bookManager = IBookManager.Stub.asInterface ( service );
try{
List<Book> list=bookManager.getBooklist();
}catch(RemoteException e){
}
}
}
@override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent=new Intent(this,BookManagerService.class);
startService(intent币狠,mConnection游两,Context.BIND_AUTO_CREATE);
}
@override
protected void onDestroy(){
unbindService(mConnection);
super.onDestroy();
}
}
總結(jié)
1、生成IBookManager.java文件(AIDL接口文件)
2漩绵、在自身中實(shí)現(xiàn)所希望別人來調(diào)用自己的方法的Binder對象(不局限于服務(wù)端客戶端贱案,只要希望其他進(jìn)程可以調(diào)用自己的方法,那么就再自己中)
Binder binder = new IBookManager.Stub(){
//實(shí)現(xiàn)接口中的方法
}
/*
實(shí)際即是接口的實(shí)現(xiàn)類
InterfaceName interface = new InterfaceName.Stub(){
//實(shí)現(xiàn)接口中的方法
}
*/
3止吐、獲取所要調(diào)用方法的AIDL接口對象(不局限于服務(wù)端或客戶端宝踪,想調(diào)用哪個進(jìn)程的AIDL方法侨糟,就以此獲取該進(jìn)程提供的對象)
InterfaceName interface = InterfaceName.Stub.asInterface(Service);
interface.getBooklist();
interface.addBook(new Book(1,"C++"));
//Service中在onServiceConnected()方法中獲取
最后
設(shè)置代理
服務(wù)端異常終止而有不知道的情況下會請求失敗瘩燥,要設(shè)置死亡代理
LinkToDeath unlinkToDeath
private IBinder.DeathRecipent mDeathRecipient = new IBinder.DeathRecipient(){
@override
public void binderDied(){ //死亡是調(diào)用
if(mBookManager == null){
return;
}
mBookManager.asBinder().unlinkToDeath(mDeathRecipient秕重,0);
mBookManager = null;
}
};
mservice = ImessageBoxManager.Stub.asInterface(binder);
binder.linkToDeath(mDeathRecipient厉膀,0);
RemoteCallbackList
用于傷處跨進(jìn)程的listener的接口溶耘,是一個泛型
public class RemoteCallbackList < E extends IInterface >
內(nèi)部有一個Map來保存所有的AIDL回調(diào)
ArrayMap <IBinder,Callback> mCallbacks = new ArrayMap<IBinder服鹅,Callback>(); IBinder key=Listener.asBinder(); Callback value=new Callback(listener凳兵,cookie);
由于跨進(jìn)程的對象是序列化和反序列化來的,所以是不同的對象企软,但是它們所指向的底層Binder對象是同一個留荔,所以解注時于服務(wù)端遍歷所有l(wèi)isnener找出相同的Binder的listener刪除即可
RemoteCallbackList是線程同步的
它并不是一個List而是接口,想遍歷要遵守以下規(guī)則:
final int N=mListenerList.beginBroadCast();
for(int i = 0; i < N;i ++){
IOnNewBookArrivedListener l = mListenerList.getBroadcastItem(i);
if (l != null){
//TODO
}
}
mListenerList.finishBroadCast();