欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android程序設(shè)計(jì)之AIDL實(shí)例詳解

 更新時(shí)間:2014年09月11日 10:50:30   投稿:shichen2014  
這篇文章主要介紹了Android程序設(shè)計(jì)的AIDL,以一個(gè)完整實(shí)例的形式較為詳細(xì)的講述了AIDL的原理及實(shí)現(xiàn)方法,需要的朋友可以參考下

通常來(lái)說(shuō),AIDL這項(xiàng)技術(shù)在我們的應(yīng)用開(kāi)發(fā)過(guò)程中并不是很常用,雖然新浪微博提供了SSO登錄,但是其原理就是使用AIDL。本文就以完整的實(shí)例形式講述了AIDL的原理及實(shí)現(xiàn)方法。

AIDL(AndRoid接口描述語(yǔ)言)是一種借口描述語(yǔ)言; 編譯器可以通過(guò)aidl文件生成一段代碼,通過(guò)預(yù)先定義的接口達(dá)到兩個(gè)進(jìn)程內(nèi)部通信進(jìn)程的目的. 如果需要在一個(gè)Activity中, 訪問(wèn)另一個(gè)Service中的某個(gè)對(duì)象, 需要先將對(duì)象轉(zhuǎn)化成 AIDL可識(shí)別的參數(shù)(可能是多個(gè)參數(shù)), 然后使用AIDL來(lái)傳遞這些參數(shù), 在消息的接收端, 使用這些參數(shù)組裝成自己需要的對(duì)象.

說(shuō)白了,AIDL就是定義一個(gè)接口,客戶端(調(diào)用端)通過(guò)bindService來(lái)與遠(yuǎn)程服務(wù)端簡(jiǎn)歷一個(gè)連接,在該連接建立時(shí)會(huì)將返回一個(gè)IBinder對(duì)象,該對(duì)象是服務(wù)端Binder的BinderProxy,在建立連接時(shí),客戶端通過(guò)asInterface函數(shù)將該BinderProxy對(duì)象包裝成本地的Proxy,并將遠(yuǎn)程服務(wù)端的BinderProxy對(duì)象賦值給Proxy類的mRemote字段,就是通過(guò)mRemote執(zhí)行遠(yuǎn)程方法調(diào)用。需要對(duì)Binder機(jī)制有更深的理解,請(qǐng)參考老羅的Android系統(tǒng)進(jìn)程間通信Binder機(jī)制在應(yīng)用程序框架層的Java接口源代碼分析。下面我們看一個(gè)AIDL實(shí)例。

AIDL接口聲明

在src目錄下創(chuàng)建一個(gè)com.example.advanceandroid.aidl包,然后在該包下創(chuàng)建一個(gè)ILogin.aidl文件,注意是創(chuàng)建文件而不是類或者接口類型。在ILogin.aidl中聲明接口,實(shí)例如下 :

package com.example.advanceandroid.aidl;
interface ILogin {
    String login();
}

注意看,接口和方法聲明都不用public,方法加入public會(huì)提示錯(cuò)誤。編寫完后如果eclipse開(kāi)啟了自動(dòng)編譯則會(huì)在gen/com.example.advanceandroid.aidl下生成一個(gè)ILogin.java類,內(nèi)容大致如下:

package com.example.advanceandroid.aidl; 
public interface ILogin extends android.os.IInterface 
{ 
  /** Local-side IPC implementation stub class. */ 
  public static abstract class Stub extends android.os.Binder implements 
      com.example.advanceandroid.aidl.ILogin 
  { 
    private static final java.lang.String DESCRIPTOR = "com.example.advanceandroid.aidl.ILogin"; 
 
    /** Construct the stub at attach it to the interface. */ 
    public Stub() 
    { 
      this.attachInterface(this, DESCRIPTOR); 
    } 
 
    /** 
     * Cast an IBinder object into an com.example.advanceandroid.aidl.ILogin 
     * interface, generating a proxy if needed. 
     */ 
    public static com.example.advanceandroid.aidl.ILogin asInterface(android.os.IBinder obj) 
    { 
      if ((obj == null)) { 
        return null; 
      } 
      android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR); 
      if (((iin != null) && (iin instanceof com.example.advanceandroid.aidl.ILogin))) { 
        return ((com.example.advanceandroid.aidl.ILogin) iin); 
      } 
      return new com.example.advanceandroid.aidl.ILogin.Stub.Proxy(obj); 
    } 
 
    @Override 
    public android.os.IBinder asBinder() 
    { 
      return this; 
    } 
 
    @Override 
    public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, 
        int flags) throws android.os.RemoteException 
    { 
      switch (code) 
      { 
        case INTERFACE_TRANSACTION: { 
          reply.writeString(DESCRIPTOR); 
          return true; 
        } 
        case TRANSACTION_login: {              // 1、登錄請(qǐng)求,執(zhí)行的是this.login(); 
          data.enforceInterface(DESCRIPTOR); 
          java.lang.String _result = this.login(); 
          reply.writeNoException(); 
          reply.writeString(_result); 
          return true; 
        } 
      } 
      return super.onTransact(code, data, reply, flags); 
    } 
 
    private static class Proxy implements com.example.advanceandroid.aidl.ILogin 
    { 
      private android.os.IBinder mRemote; 
 
      Proxy(android.os.IBinder remote) 
      { 
        mRemote = remote; 
      } 
 
      @Override 
      public android.os.IBinder asBinder() 
      { 
        return mRemote; 
      } 
 
      public java.lang.String getInterfaceDescriptor() 
      { 
        return DESCRIPTOR; 
      } 
 
      @Override 
      public java.lang.String login() throws android.os.RemoteException        // 2、Proxy中的login,通過(guò)Binder機(jī)制實(shí)現(xiàn)IPC 
      { 
        android.os.Parcel _data = android.os.Parcel.obtain(); 
        android.os.Parcel _reply = android.os.Parcel.obtain(); 
        java.lang.String _result; 
        try { 
          _data.writeInterfaceToken(DESCRIPTOR); 
          mRemote.transact(Stub.TRANSACTION_login, _data, _reply, 0); 
          _reply.readException(); 
          _result = _reply.readString(); 
        } finally { 
          _reply.recycle(); 
          _data.recycle(); 
        } 
        return _result; 
      } 
    } 
 
    static final int TRANSACTION_login = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0); 
  } 
  public java.lang.String login() throws android.os.RemoteException; 
} 

可以看到,該類中自動(dòng)生成了ILogin接口,該接口中又一個(gè)login()函數(shù)。但最重要的是里面生成了一個(gè)Stub類,該類集成子Binder類,并且實(shí)現(xiàn)了ILogin接口。Stub里面最重要的就是asInterface()這個(gè)函數(shù),在這個(gè)函數(shù)中會(huì)判斷obj參數(shù)的類型,如果是該obj是本地的接口類似,則認(rèn)為不是IPC,會(huì)將該obj轉(zhuǎn)換成ILogin類型;否則會(huì)通過(guò)自動(dòng)生成的另一個(gè)內(nèi)部類Proxy來(lái)包裝obj,將其賦值給Proxy中的mRemote屬性。Proxy類也實(shí)現(xiàn)了ILogin接口,在login()函數(shù)中,Proxy將通過(guò)Binder機(jī)制向服務(wù)端傳遞請(qǐng)求和數(shù)據(jù),如上面代碼中的注釋2。這是客戶端的工作算是完成了。

服務(wù)端AIDL接口

服務(wù)端也需要在相同的包下創(chuàng)建同名的aidl文件,我們直接將客戶端的com.example.advanceandroid.aidl包下的ILogin.aidl拷貝到服務(wù)端即可,如果用到了自定義的類型,那么該自定義類型也需要在客戶端、服務(wù)端都有??截愅阛idl后,在服務(wù)端程序中也會(huì)在gen中生成對(duì)應(yīng)的ILogin.java文件,內(nèi)容同客戶端一樣。這里的重點(diǎn)我們要看onTransact函數(shù),即上述代碼中的注釋1處,可以看到,在case TRANSACTION_login處執(zhí)行了this.login()函數(shù),意思是當(dāng)接收到客戶端的TRANSACTION_login請(qǐng)求時(shí),執(zhí)行this.login()函數(shù),通過(guò)客戶端的分析我們知道,當(dāng)我們調(diào)用login()時(shí)實(shí)際上就是通過(guò)mRemote向服務(wù)端提交了一個(gè)TRANSACTION_login請(qǐng)求,因此就兩端通過(guò)Binder機(jī)制就對(duì)接上了,我們可以簡(jiǎn)單的理解為C/S模式。

服務(wù)端還沒(méi)有完,最重要的一步時(shí)建立一個(gè)Service,內(nèi)容大致如下 :

/** 
* AIDL服務(wù)端接口,LoginStubImpl實(shí)現(xiàn)了ILogin接口. 
* 
* @author mrsimple 
*/ 
public class LoginService extends Service { 
 
  /** 
   * 
   */ 
  IBinder mBinder = new LoginStubImpl(); 
 
  /** 
   * @author mrsimple 
   */ 
  class LoginStubImpl extends Stub { 
    @Override 
    public String login() throws RemoteException { 
      return "這是從 " + this.getClass().getName() + " 返回的字符串"; 
    } 
  } 
 
  /* 
   * 返回Binder實(shí)例,即實(shí)現(xiàn)了ILogin接口的Stub的子類,這里為L(zhǎng)oginStubImpl 
   * [url=home.php?mod=space&uid=133757]@see[/url] android.app.Service#onBind(android.content.Intent) 
   */ 
  @Override 
  public IBinder onBind(Intent intent) { 
    return mBinder; 
  } 
} 

該Service我們這里命名為L(zhǎng)oginService,繼承自Service,然后建一個(gè)名為L(zhǎng)oginServiceImpl的內(nèi)部類,該類繼承自自動(dòng)生成的Stub,然后實(shí)現(xiàn)login()方法。在LoginService中聲明一個(gè)IBinder字段mBinder :

IBinder mBinder = new LoginStubImpl();

并且在LoginService的onBind函數(shù)中將mBinder對(duì)象返回。即在客戶端建立與服務(wù)端的連接時(shí),會(huì)調(diào)用onBind方法將mBinder對(duì)象返回,在客戶端的ServiceConnection類的onServiceConnected函數(shù)中得到的對(duì)象IBinder就是經(jīng)過(guò)BinderProxy包裝的LoginService中的mBinder對(duì)象。因此在服務(wù)端中的onTransact中調(diào)用的this.login()函數(shù)實(shí)際上就是調(diào)用的LoginStubImpl中的login()函數(shù)。

在服務(wù)端程序的AndroidManifest.xml中注冊(cè)LoginService,如下 :

<!-- aidl server service --> 
<service android:name="com.example.advanceandroid.aidl.LoginService" > 
    <intent-filter> 
      <action android:name="com.example.advanceandroid.aidl.LoginService" /> 
    </intent-filter> 
</service> 

客戶端建立連接

在Activity中加入如下代碼 :

ServiceConnection mLoginConnection = new ServiceConnection() { 
 
    @Override 
    public void onServiceDisconnected(ComponentName name) { 
      Log.d("", "### aidl disconnected."); 
    } 
 
    @Override 
    public void onServiceConnected(ComponentName name, IBinder service) { 
      Log.d("", "### aidl onServiceConnected.   service : " + service.getClass().getName()); 
 
      ILogin login = Stub.asInterface(service); 
      Log.d("", "### after asInterface : " + login.getClass().getName()); 
      try { 
        Log.d("", "### login : " + login.login()); 
        // Toast.makeText(MainActivity.this, "onServiceConnected : " + 
        // login.login(), 
        // Toast.LENGTH_SHORT).show(); 
      } catch (RemoteException e) { 
        e.printStackTrace(); 
      } 
    } 
  }; 
 
  @Override 
  protected void onResume() { 
    super.onResume(); 
 
    // 服務(wù)端的action 
    Intent aidlIntent = new Intent("com.example.advanceandroid.aidl.LoginService"); 
    bindService(aidlIntent, mLoginConnection, Context.BIND_AUTO_CREATE); 
  } 
 
  @Override 
  protected void onStop() { 
    super.onStop(); 
    // unbind 
    unbindService(mLoginConnection); 
  } 

運(yùn)行

先運(yùn)行服務(wù)端程序,然后在啟動(dòng)客戶端程序,可以看到客戶端輸出如下Log:

09-02 10:40:54.662: D/(9589): ### aidl onServiceConnected.   service : android.os.BinderProxy 
09-02 10:40:54.662: D/(9589): ### after asInterface : com.example.advanceandroid.aidl.ILogin$Stub$Proxy 
09-02 10:40:54.662: D/(9589): ### login : 這是從 com.example.advanceandroid.aidl.LoginService$LoginStubImpl 返回的字符串 

可以看淡onServiceConnected(ComponentName name, IBinder service)中的service對(duì)象是BinderProxy類型,經(jīng)過(guò)asInterface轉(zhuǎn)換后被包裝成了Proxy類型,但是調(diào)用的時(shí)候,執(zhí)行的是服務(wù)端LoginStubImpl中的login()函數(shù)。因此,LoginStubImpl實(shí)例mBinder被服務(wù)端包裝成BinderProxy類型,再經(jīng)過(guò)客戶端的Proxy進(jìn)行包裝,通過(guò)Binder機(jī)制進(jìn)行數(shù)據(jù)傳輸,實(shí)現(xiàn)IPC。

希望本文所述對(duì)大家進(jìn)一步深入掌握Android程序設(shè)計(jì)有所幫助。

相關(guān)文章

  • Android仿微信調(diào)用第三方地圖應(yīng)用導(dǎo)航(高德、百度、騰訊)

    Android仿微信調(diào)用第三方地圖應(yīng)用導(dǎo)航(高德、百度、騰訊)

    這篇文章主要介紹了Android仿微信調(diào)用第三方地圖應(yīng)用導(dǎo)航,包括高德、百度、騰訊地圖,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • Android實(shí)現(xiàn)掃一掃識(shí)別數(shù)字功能

    Android實(shí)現(xiàn)掃一掃識(shí)別數(shù)字功能

    這篇文章主要介紹了Android實(shí)現(xiàn)掃一掃識(shí)別數(shù)字功能,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2017-09-09
  • Android串口通信之串口讀寫實(shí)例

    Android串口通信之串口讀寫實(shí)例

    這篇文章主要為大家詳細(xì)介紹了Android串口通信之串口讀寫實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-04-04
  • Android自定義APP全局懸浮按鈕

    Android自定義APP全局懸浮按鈕

    這篇文章主要為大家詳細(xì)介紹了Android自定義APP全局懸浮按鈕,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • Android中數(shù)據(jù)解析的五種方式

    Android中數(shù)據(jù)解析的五種方式

    今天小編就為大家分享一篇關(guān)于Android中數(shù)據(jù)解析的五種方式,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2019-02-02
  • Android基于Glide v4.x的圖片加載進(jìn)度監(jiān)聽(tīng)

    Android基于Glide v4.x的圖片加載進(jìn)度監(jiān)聽(tīng)

    本篇文章主要介紹了基于Glide v4.x的圖片加載進(jìn)度監(jiān)聽(tīng)的示例代碼,具有一定的參考價(jià)值,有興趣的可以了解一下
    2017-08-08
  • Android Studio 中運(yùn)行 groovy 程序的方法圖文詳解

    Android Studio 中運(yùn)行 groovy 程序的方法圖文詳解

    這篇文章主要介紹了Android Studio 中 運(yùn)行 groovy 程序的方法,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-03-03
  • Android自定義TextView跑馬燈效果

    Android自定義TextView跑馬燈效果

    這篇文章主要為大家詳細(xì)介紹了Android自定義TextView跑馬燈效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • Android實(shí)現(xiàn)帶磁性的懸浮窗體效果

    Android實(shí)現(xiàn)帶磁性的懸浮窗體效果

    這篇文章主要介紹了Android實(shí)現(xiàn)帶磁性的懸浮窗體效果,涉及Android針對(duì)窗體的動(dòng)態(tài)操作相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2016-07-07
  • Android系統(tǒng)添加自己寫的工具

    Android系統(tǒng)添加自己寫的工具

    今天小編就為大家分享一篇關(guān)于Android系統(tǒng)添加自己寫的工具的文章,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-10-10

最新評(píng)論