日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android Service和Binder、AIDL

發布時間:2024/4/15 Android 31 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android Service和Binder、AIDL 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

為什么80%的碼農都做不了架構師?>>> ??

Android Service和Binder、AIDL

人收藏此文章,?關注此文章發表于3個月前 , 已有 206次閱讀 共 個評論? 人收藏此文章

1.首先理解service的作用和生命周期,

由于activity如果切換,那么他就不再運行,那么我們想在玩游戲的時候聽播放器中的音樂,activity就應運而生了,這是最常見的一種場景,同時service由于它的優先級比較高,不容易被回收,而且是獨立進程,不會阻塞UI線程,因此,可以用來處理一些比較費時的任務。

service起于startService(),終于stopService,如果沒有調用stopService,那么,即使調用者結束了,該service也一直存在。

也可以通過bindService來綁定service,unBindService分開并結束service。如果bind的時候沒有啟動service,那么它會調用service的create方法啟動。

多個程序可同時bind同一個service,只有他們都unbind了,這個service就會自動結束。如果有部分unbinder的時候調用stopservice,這個stop也會等到它們全部結束的時候才真的結束。

?

2.service的種類,

由于adroid的獨特的線程模型,service被同一個apk調用和不同apk調用原理是不同的,因此分成以下兩種:

(1). 本地服務(Local Service):說白了就是在同一個apk內被調用。

(2). 遠程服務(Remote Sercie):被另外一個apk調用。

?

3.涉及到的內容:

由于android的進程模型,不同的apk不能共用數據,因此如果有需要的話需要通過進程間通訊完成。

進程間通訊(ipc)涉及到三個部分:客戶端(調用方),傳遞數據,服務端(被調用方)。

android的數據傳遞類似于RPC過程,采用aidl的方式傳遞,考慮到效率的原因,沒有用java內置的serializable,而是采用原始數據拆分組裝的parcel方式。

?

4.場景假設:

一個service提供產生Student數據的功能,作為服務端,一個apk中的activity想獲取另一個apk中的一個學生。根據上邊的分析,service即使服務端,activity即使客戶端,student是要傳輸的數據,要被包裝成pacel傳遞。

?

5.根據android思想,理想的傳遞過程:

student提供拆分成parcel的writeToParcel()方法和根據Parcel組裝的構造函數Student(Pacel p);

一個bissiness Service(bizSvc)用來實現邏輯功能。

一個android service(adSvc)調用bizService提供功能。

一個MyBinder負責service描述符與bizService的映射(本地服務時供queryLocalInterface查找)和與客戶端交互。(如果被本地調用,就查找到相應的service直接操作,不用遠程通訊,如果被遠程調用,就得負責與遠程通訊)

一個Proxy負責在客戶端提供transact()方法發送數據。

?

具體的調用過程如下:

調用的時候,客戶端首先調用bindService(new Intent ("abc"), serviceConnection,Context.BIND_AUTO_CREATE);激活serviceConnection的onServiceConnected方法,在此方法中獲取到一個binder(注:clientBinder,系統給我們的一個與遠程進行通信的binder,不是我們剛才自己實現的),此binder能夠查找系統中注冊的service,如果沒有查找到該Service,那么可認定該service是從其他apk獲得的,就創建一個此service的靜態代理類Proxy,否則,就把這個service返回,供客戶端調用。

?

服務端收到這個Intent后,激活adSvc的onBind方法,創建一個MyBinder返回。之后,這個MyBinder負責與客戶端的Proxy通信。

?

之后,客戶端要調用service的方法,可直接調用(本地)或者通過代理調用(遠程),這個Proxy調用clientBinder的transact方法,參數為要掉用的方法(TRANSACRION_XX),發送的參數(_data),接受的返回值(_reply),把消息傳送給服務端。

服務端收到消息后,調用MyBinder的onTransac方法,根據Proxy傳遞過來參數,調用bizService不同的方法,并把產生的值組裝成Parcel發送回去。之后客戶端Proxy會自動調用sudent的相關方法,把數據重新組裝,進行下一步處理。

?

理想的代碼如下(這是理想的代碼,經過拆分,但是不符合android的aidl規范,不能運行,見下文分析):

客戶端代碼:

final IMyBizService bizSvc;

bindService(new Intent("abc"),new ServiceConnection(){

@Override

public void onServiceConnected(ComponentName name, IBinder clientBinder) {

if ((binder == null)) {

return null;

}

IInterface iin = ?clientBinder.queryLocalInterface(DESCRIPTOR);//如果是查找本地有,就直接返回

if (((iin != null) && (iin instanceof IMyService))) {

return ((IMyService) iin);

}

return new Proxy(obj);//否則采用遠程代理

?

}

}, Context.BIND_AUTO_CREATE);

?

Proxy.java:

public class Proxy implements IMyBizService {

private IBinder mRemote;

private String description;

?

public Proxy(IBinder binder,String description) {

super();

this.mRemote=binder;

this.description=description;

}

?

@Override

public IBinder asBinder() {

return mRemote;

}

?

public Student getStudent() throws android.os.RemoteException {

android.os.Parcel _data = android.os.Parcel.obtain();

android.os.Parcel _reply = android.os.Parcel.obtain();

Student _result;

try {

_data.writeInterfaceToken(description);

mRemote.transact(MyBinder.TRANSACTION_getStudent, _data,_reply, 0);//發送參數給服務端

_reply.readException();

if ((0 != _reply.readInt())) {

_result = new Student(_reply);//接受參數并返回

} else {

_result = null;

}

} finally {

_reply.recycle();

_data.recycle();

}

return _result;

}

?

}

?

?

服務端代碼:

MyService.java:

public class MyService extends Service {

@Override

public IBinder onBind(Intent intent) {

MyBizServiceImpl svc=new MyBizServiceImpl();

MyBinder binder= new MyBinder(new MyBizServiceImpl(),intent.getAction());

svc.setBinder(binder);//(此處有點繞,binder方法需要bizService作為參數實現綁定和調用,而bizSvc需要這個Binder作為參數,用來實現asBind方法,形成了一個環);

return binder;

}

}

?

MyBinder.java:

public class MyBinder extends Binder {

static final int TRANSACTION_getStudent = (IBinder.FIRST_CALL_TRANSACTION + 1);

private IInterface intf;

private String descripter;

?

public MyBinder(IInterface intf, String descripter) {

super();

this.intf=intf;

this.descripter=descripter;

this.attachInterface(intf, descripter);

}

?

@Override

public boolean onTransact(int code, Parcel data,Parcel reply, int flags)throws RemoteException {

IMyBizService myService=(IMyBizService) this.intf;

switch (code) {

case INTERFACE_TRANSACTION: {

reply.writeString(descripter);

return true;

}

case TRANSACTION_getStudent: {

data.enforceInterface(descripter);

Student _result = myService.getStudent();

reply.writeNoException();

if ((_result != null)) {

reply.writeInt(1);

_result.writeToParcel(reply,Parcelable.PARCELABLE_WRITE_RETURN_VALUE);//發送數據給客戶端

} else {

reply.writeInt(0);

}

return true;

}

}

return super.onTransact(code, data, reply, flags);

}

}

?

MyBizService.java:

public class MyBizServiceImpl implements IMyBizService {

private IBinder binder;

public void setBinder(IBinder binder){

this.binder=binder;

}

@Override

public IBinder asBinder() {

return binder;

}

?

@Override

public Student getStudent() throws RemoteException {

Student st = new Student();

st.setName("devil");

?

return st;

}

?

}

?

5 aidl對這個機制的改進:

第4步的代碼不能執行,只能作為理解原理和閱讀代碼用。因為為了安全和效率,aidl的關鍵類自動生成,不能修改,同時android對aidl的保護也使得需要一些額外的檢查。

對比以上代碼,android自動生成的代碼有以下幾個方面不同:

1).MyBinder,IMyBizService,和Proxy通通根據IMyBizService.aidl文件自動生成為IMyBizService.java文件。具體關系如下:

MyBinder和IMyBizService被合并成一個Stub類作為IMyBizService的內部類(直接導致this被濫用,但同時實現了這兩個,那么相關方法直接用this就行了,不用形成前文所說的環了)。這個類負責接與客戶端交互。而且還有一個額外靜態方法asInterface供客戶端在引起connection的onConnected時調用,實現根據傳入的clientBinder判斷本次調用是本地調用還是遠程調用,從而確定是否采用Proxy,這樣這個方法既然已經實現了,那么我們直接用就行了,無須再寫一遍。具體IMyBizService的實現,只需繼承Stub類即可。這樣,只需在adService中創建MyBinder的時候,也就同時創建了MyBizService.

Proxy作為Stub的內部類(真糾結。。),被asInterface調用,從而進一步被客戶端調用。

2).由于自動生成的IMyBizService.java中的Stub的onTransact利用了Student類中的一個CREATE內部域作為轉換器,我們必須在Student類中添加并實現這個域,注意:域名必須是CREATE大寫。(這個實現有點囧,可能是因為在自動生成文件的時候無法確定這個轉換方法的寫法)。

?

6 這個模型一些特例

1).如果本service只作為本地調用,可無須aidl相關操作,服務端的binder直接像stub那樣,即使binder又是service,這樣,調用方(如Activity)由于與該service在同一個apk中,ServiceConnection中獲取的clientBind就是service返回的那個myBinder,這樣的話,直接強轉即可得到IMyBizService.不用調用queryLocalInterface()方法再查找一次。

2).如果不用額外調用service的其他方法,只是啟動一個service的話,那么直接用startService即可,無須bind。

?

7.對這個模型的感受

這個模型有點囧,作者本來想讓碼工少寫點代碼,結果理解起來結構感覺亂糟糟的,而且沒有一條主線,內部類各種相互調,客戶端服務端不分,太不給力了。除非使用者之前有良好的rpc功底和代碼運用能力,否則,要實現自己定制的功能比以前還得小心,要不就是對著示例代碼照貓畫虎,調試起來就是一種折磨。(一家之言,有理解不對的地方多多提醒,文中示例參考了http://4225953-163-com.javaeye.com/blog/792997一文中的示例)

?

8.附帶標準aidl實現的完整過程:

1)實現要傳遞的數據結構和aidl(student.java和student.aidl)

2)實現相關的service的aidl(IMyBizService.aidl)

2)如果是eclipse,在1,2完成的情況下,會自動在gen目錄下生成IMyBizService.java文件,如果非eclispse,則需要用sdk/platfroms/{version}/tools/aidl工具生成。

3)在myService中集成Stub類,實現IMyBizServic的方法,并作為binder在onBind方法中返回。

4)把IBizInterface.java和Student.java考進客戶端工程(aidl文件安不需要,并且包名不能改變),客戶端調用Stub的asInterface方法獲取到相關的接口代理

5)利用接口代理調用各種方法

6)對于以上各個步驟中的具體操作,網上到處都是。

人收藏此文章,?關注此文章發表于3個月前 , 已有 206次閱讀 共 個評論? 人收藏此文章

1.首先理解service的作用和生命周期,

由于activity如果切換,那么他就不再運行,那么我們想在玩游戲的時候聽播放器中的音樂,activity就應運而生了,這是最常見的一種場景,同時service由于它的優先級比較高,不容易被回收,而且是獨立進程,不會阻塞UI線程,因此,可以用來處理一些比較費時的任務。

service起于startService(),終于stopService,如果沒有調用stopService,那么,即使調用者結束了,該service也一直存在。

也可以通過bindService來綁定service,unBindService分開并結束service。如果bind的時候沒有啟動service,那么它會調用service的create方法啟動。

多個程序可同時bind同一個service,只有他們都unbind了,這個service就會自動結束。如果有部分unbinder的時候調用stopservice,這個stop也會等到它們全部結束的時候才真的結束。

?

2.service的種類,

由于adroid的獨特的線程模型,service被同一個apk調用和不同apk調用原理是不同的,因此分成以下兩種:

(1). 本地服務(Local Service):說白了就是在同一個apk內被調用。

(2). 遠程服務(Remote Sercie):被另外一個apk調用。

?

3.涉及到的內容:

由于android的進程模型,不同的apk不能共用數據,因此如果有需要的話需要通過進程間通訊完成。

進程間通訊(ipc)涉及到三個部分:客戶端(調用方),傳遞數據,服務端(被調用方)。

android的數據傳遞類似于RPC過程,采用aidl的方式傳遞,考慮到效率的原因,沒有用java內置的serializable,而是采用原始數據拆分組裝的parcel方式。

?

4.場景假設:

一個service提供產生Student數據的功能,作為服務端,一個apk中的activity想獲取另一個apk中的一個學生。根據上邊的分析,service即使服務端,activity即使客戶端,student是要傳輸的數據,要被包裝成pacel傳遞。

?

5.根據android思想,理想的傳遞過程:

student提供拆分成parcel的writeToParcel()方法和根據Parcel組裝的構造函數Student(Pacel p);

一個bissiness Service(bizSvc)用來實現邏輯功能。

一個android service(adSvc)調用bizService提供功能。

一個MyBinder負責service描述符與bizService的映射(本地服務時供queryLocalInterface查找)和與客戶端交互。(如果被本地調用,就查找到相應的service直接操作,不用遠程通訊,如果被遠程調用,就得負責與遠程通訊)

一個Proxy負責在客戶端提供transact()方法發送數據。

?

具體的調用過程如下:

調用的時候,客戶端首先調用bindService(new Intent ("abc"), serviceConnection,Context.BIND_AUTO_CREATE);激活serviceConnection的onServiceConnected方法,在此方法中獲取到一個binder(注:clientBinder,系統給我們的一個與遠程進行通信的binder,不是我們剛才自己實現的),此binder能夠查找系統中注冊的service,如果沒有查找到該Service,那么可認定該service是從其他apk獲得的,就創建一個此service的靜態代理類Proxy,否則,就把這個service返回,供客戶端調用。

?

服務端收到這個Intent后,激活adSvc的onBind方法,創建一個MyBinder返回。之后,這個MyBinder負責與客戶端的Proxy通信。

?

之后,客戶端要調用service的方法,可直接調用(本地)或者通過代理調用(遠程),這個Proxy調用clientBinder的transact方法,參數為要掉用的方法(TRANSACRION_XX),發送的參數(_data),接受的返回值(_reply),把消息傳送給服務端。

服務端收到消息后,調用MyBinder的onTransac方法,根據Proxy傳遞過來參數,調用bizService不同的方法,并把產生的值組裝成Parcel發送回去。之后客戶端Proxy會自動調用sudent的相關方法,把數據重新組裝,進行下一步處理。

?

理想的代碼如下(這是理想的代碼,經過拆分,但是不符合android的aidl規范,不能運行,見下文分析):

客戶端代碼:

final IMyBizService bizSvc;

bindService(new Intent("abc"),new ServiceConnection(){

@Override

public void onServiceConnected(ComponentName name, IBinder clientBinder) {

if ((binder == null)) {

return null;

}

IInterface iin = ?clientBinder.queryLocalInterface(DESCRIPTOR);//如果是查找本地有,就直接返回

if (((iin != null) && (iin instanceof IMyService))) {

return ((IMyService) iin);

}

return new Proxy(obj);//否則采用遠程代理

?

}

}, Context.BIND_AUTO_CREATE);

?

Proxy.java:

public class Proxy implements IMyBizService {

private IBinder mRemote;

private String description;

?

public Proxy(IBinder binder,String description) {

super();

this.mRemote=binder;

this.description=description;

}

?

@Override

public IBinder asBinder() {

return mRemote;

}

?

public Student getStudent() throws android.os.RemoteException {

android.os.Parcel _data = android.os.Parcel.obtain();

android.os.Parcel _reply = android.os.Parcel.obtain();

Student _result;

try {

_data.writeInterfaceToken(description);

mRemote.transact(MyBinder.TRANSACTION_getStudent, _data,_reply, 0);//發送參數給服務端

_reply.readException();

if ((0 != _reply.readInt())) {

_result = new Student(_reply);//接受參數并返回

} else {

_result = null;

}

} finally {

_reply.recycle();

_data.recycle();

}

return _result;

}

?

}

?

?

服務端代碼:

MyService.java:

public class MyService extends Service {

@Override

public IBinder onBind(Intent intent) {

MyBizServiceImpl svc=new MyBizServiceImpl();

MyBinder binder= new MyBinder(new MyBizServiceImpl(),intent.getAction());

svc.setBinder(binder);//(此處有點繞,binder方法需要bizService作為參數實現綁定和調用,而bizSvc需要這個Binder作為參數,用來實現asBind方法,形成了一個環);

return binder;

}

}

?

MyBinder.java:

public class MyBinder extends Binder {

static final int TRANSACTION_getStudent = (IBinder.FIRST_CALL_TRANSACTION + 1);

private IInterface intf;

private String descripter;

?

public MyBinder(IInterface intf, String descripter) {

super();

this.intf=intf;

this.descripter=descripter;

this.attachInterface(intf, descripter);

}

?

@Override

public boolean onTransact(int code, Parcel data,Parcel reply, int flags)throws RemoteException {

IMyBizService myService=(IMyBizService) this.intf;

switch (code) {

case INTERFACE_TRANSACTION: {

reply.writeString(descripter);

return true;

}

case TRANSACTION_getStudent: {

data.enforceInterface(descripter);

Student _result = myService.getStudent();

reply.writeNoException();

if ((_result != null)) {

reply.writeInt(1);

_result.writeToParcel(reply,Parcelable.PARCELABLE_WRITE_RETURN_VALUE);//發送數據給客戶端

} else {

reply.writeInt(0);

}

return true;

}

}

return super.onTransact(code, data, reply, flags);

}

}

?

MyBizService.java:

public class MyBizServiceImpl implements IMyBizService {

private IBinder binder;

public void setBinder(IBinder binder){

this.binder=binder;

}

@Override

public IBinder asBinder() {

return binder;

}

?

@Override

public Student getStudent() throws RemoteException {

Student st = new Student();

st.setName("devil");

?

return st;

}

?

}

?

5 aidl對這個機制的改進:

第4步的代碼不能執行,只能作為理解原理和閱讀代碼用。因為為了安全和效率,aidl的關鍵類自動生成,不能修改,同時android對aidl的保護也使得需要一些額外的檢查。

對比以上代碼,android自動生成的代碼有以下幾個方面不同:

1).MyBinder,IMyBizService,和Proxy通通根據IMyBizService.aidl文件自動生成為IMyBizService.java文件。具體關系如下:

MyBinder和IMyBizService被合并成一個Stub類作為IMyBizService的內部類(直接導致this被濫用,但同時實現了這兩個,那么相關方法直接用this就行了,不用形成前文所說的環了)。這個類負責接與客戶端交互。而且還有一個額外靜態方法asInterface供客戶端在引起connection的onConnected時調用,實現根據傳入的clientBinder判斷本次調用是本地調用還是遠程調用,從而確定是否采用Proxy,這樣這個方法既然已經實現了,那么我們直接用就行了,無須再寫一遍。具體IMyBizService的實現,只需繼承Stub類即可。這樣,只需在adService中創建MyBinder的時候,也就同時創建了MyBizService.

Proxy作為Stub的內部類(真糾結。。),被asInterface調用,從而進一步被客戶端調用。

2).由于自動生成的IMyBizService.java中的Stub的onTransact利用了Student類中的一個CREATE內部域作為轉換器,我們必須在Student類中添加并實現這個域,注意:域名必須是CREATE大寫。(這個實現有點囧,可能是因為在自動生成文件的時候無法確定這個轉換方法的寫法)。

?

6 這個模型一些特例

1).如果本service只作為本地調用,可無須aidl相關操作,服務端的binder直接像stub那樣,即使binder又是service,這樣,調用方(如Activity)由于與該service在同一個apk中,ServiceConnection中獲取的clientBind就是service返回的那個myBinder,這樣的話,直接強轉即可得到IMyBizService.不用調用queryLocalInterface()方法再查找一次。

2).如果不用額外調用service的其他方法,只是啟動一個service的話,那么直接用startService即可,無須bind。

?

7.對這個模型的感受

這個模型有點囧,作者本來想讓碼工少寫點代碼,結果理解起來結構感覺亂糟糟的,而且沒有一條主線,內部類各種相互調,客戶端服務端不分,太不給力了。除非使用者之前有良好的rpc功底和代碼運用能力,否則,要實現自己定制的功能比以前還得小心,要不就是對著示例代碼照貓畫虎,調試起來就是一種折磨。(一家之言,有理解不對的地方多多提醒,文中示例參考了http://4225953-163-com.javaeye.com/blog/792997一文中的示例)

?

8.附帶標準aidl實現的完整過程:

1)實現要傳遞的數據結構和aidl(student.java和student.aidl)

2)實現相關的service的aidl(IMyBizService.aidl)

2)如果是eclipse,在1,2完成的情況下,會自動在gen目錄下生成IMyBizService.java文件,如果非eclispse,則需要用sdk/platfroms/{version}/tools/aidl工具生成。

3)在myService中集成Stub類,實現IMyBizServic的方法,并作為binder在onBind方法中返回。

4)把IBizInterface.java和Student.java考進客戶端工程(aidl文件安不需要,并且包名不能改變),客戶端調用Stub的asInterface方法獲取到相關的接口代理

5)利用接口代理調用各種方法

6)對于以上各個步驟中的具體操作,網上到處都是。

轉載于:https://my.oschina.net/ykai/blog/39207

總結

以上是生活随笔為你收集整理的Android Service和Binder、AIDL的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。