安卓AIDL

2020-03-11  本文已影响0人  VanHoan

Android Interface Definition Language(AIDL)

AIDL与您可能使用过的其他接口语言(IDL)类似。可以利用它定义客户端与服务端均认可的编程接口,以便二者能够使用进程间通信(IPC)进行相互通信。在Android中,一个进程通常无法访问另一个进程的内存。因此为了进行通信,进程需要将对象分解成可供操作系统理解的原语,并将其编组为可供应用操作的对应。

<p style="display: block; font-size: 14px; background: #e1f5fe; color: #01579b; padding: 16px 24px">注意:只有在需要不同应用的客户端通过IPC方式访问服务,并且希望在服务中进行多线程处理时,才有必要使用AIDL。如果无需要跨进程执行并发IPC,则应用通过实现Binder来创建接口。仅想执行IPC,不需要多线程处理,请使用Messenger来实现接口。</p>

简单一点的说:

定义AIDL接口

在 .aidl 文件中使用 Java 编程语言的语法定义 AIDL 接口,然后将其保存至应用的源代码(在 src/ 目录中)内,这类应用会托管服务或与服务进行绑定。

在构建每个包含 .aidl 文件的应用时,Android SDK 工具会生成基于该 .aidl 文件的 IBinder 接口,并将其保存到项目的 gen/ 目录中。服务必须视情况实现 IBinder 接口。然后,客户端应用便可绑定到该服务,并调用 IBinder 中的方法来执行 IPC。

执行以下步骤创建AIDL绑定服务:

  1. 创建.aidl文件:些文件定义了带有方法签名的接口
  2. 实现接口:Android SDK中的aidl命令行工具会生成一个基于Java的抽象实现。接口中拥有一个名为Stub的内部抽象类,用于扩展Binder类并实现AIDL中的方法。服务提供者必须实现这个Stub,用于对外提供服务。
  3. 向客户端公开接口:实现Service并重写onBind(),从而返回第二步中的实现类。

<p style="display: block; font-size: 14px; background: #feefe3; color: #bf360c; padding: 16px 24px">注意:如果您在首次发布 AIDL 接口后对其进行更改,则每次更改必须保持向后兼容性,以免中断其他应用使用您的服务。换言之,由于只有在将您的 .aidl 文件复制到其他应用后,才能使其访问服务接口,因而您必须保留对原始接口的支持。</p>

1. 创建.aidl文件

AIDL默认支持以下数据类型:

<p style="display: block; font-size: 14px; background: #e1f5fe; color: #01579b; padding: 16px 16px">
即使您在与接口相同的包内定义上方未列出的附加类型,亦须为其各自加入一条 import 语句。</-p>

定义服务接口时,需要注意:

<font color=BrulyWood>下面列出inoutinoutoneway关键字对AIDL实现的影响:</font>

实际上不管是有没有oneway修饰的AIDL方法,在服务端实现时都是运行在Binder线程池中的,只不过会没有oneway修饰的方法实现,在客户端会阻塞在当前调用线程直到返回才能继续执行下一个操作。

// flags = 0
boolean _status = mRemote.transact(
              Stub.TRANSACTION_fetchTodayCallback,
              _data, _reply,
              0);

// flags = FLAG_ONEWAY
boolean _status = mRemote.transact(
              Stub.TRANSACTION_asyncFetchTodayCallback,
              _data, null,
              android.os.IBinder.FLAG_ONEWAY);

2. 实现接口

当构建应用时,Android SDK 工具会生成以 .aidl 文件命名的 .java 接口文件。生成的接口包含一个名为 Stub 的子类(例如,YourInterface.Stub),该子类是其父接口的抽象实现,并且会声明 .aidl 文件中的所有方法。

<p style="display: block; font-size: 14px; background: #e1f5fe; color: #01579b; padding: 16px 24px">Stub还定义了几个辅助方法,其中最值得注意的是asInterface(),该方法会接收IBinder(通过是传递给客户端onServiceConnected回调方法的参数),并返回Stub接口的实例。</p>

在实现 AIDL 接口时,您应注意遵守以下规则:

3. 向客户端公开接口

通过实现Service中的onBind方法返回上一步实现的接口,完成对外提供服务的能力

如果客户端和服务在不同的应用内,那么客户端应用的里必须包含访问服务接口的AIDL抽象实现,或者由服务提供方提供相应的访问SDK。

当客户端在onServiceConnected()回调中收到IBinder时,通过YourServiceInterface.Stub.asInterface(service)返回YourServiceInterface类型的实例。如果在同一进程,该实例与Service中onBind返回的是同一实例,否则返回的是一个远程代理实例YourServiceInterface.Stube.Proxy

通过IPC传递对象

可以通过 IPC 接口,将某个类从一个进程发送至另一个进程。不过,您必须确保 IPC 通道的另一端可使用该类的代码,并且该类必须支持 Parcelable 接口。支持 Parcelable 接口很重要,因为 Android 系统能通过该接口将对象分解成可编组至各进程的原语。

如要创建支持 Parcelable 协议的类,您必须执行以下操作:

  1. 让您的类实现 Parcelable 接口。
  2. 实现 writeToParcel,它会获取对象的当前状态并将其写入 Parcel。
  3. 为您的类添加 CREATOR 静态字段,该字段是实现 Parcelable.Creator 接口的对象。
  4. 最后,创建声明 Parcelable 类的 .aidl 文件(遵照下文 Rect.aidl 文件所示步骤)。如果您使用的是自定义编译进程,请勿在您的构建中添加 .aidl 文件。此 .aidl 文件与 C 语言中的头文件类似,并未经过编译。
package android.graphics;

// Declare Rect so AIDL can find it and
// knows that it implements
// the parcelable protocol.
parcelable Rect;

带软件包参数(包含 Parcelable 类型)的方法

如果您的 AIDL 接口包含接收软件包作为参数(预计包含 Parcelable 类型)的方法,则在尝试从软件包读取之前,请务必通过调用 Bundle.setClassLoader(ClassLoader) 设置软件包的类加载器。否则,即使您在应用中正确定义 Parcelable 类型,也会遇到 ClassNotFoundException。例如,

如果您有 .aidl 文件:

// IRectInsideBundle.aidl
package com.example.android;

/** Example service interface */
interface IRectInsideBundle {
    /** Rect parcelable is stored in the bundle with key "rect" */
    void saveRect(in Bundle bundle);
}

如下方实现所示,在读取 Rect 之前,ClassLoader 已在 Bundle 中完成显式设置

private final IRectInsideBundle.Stub binder = new IRectInsideBundle.Stub() {
    public void saveRect(Bundle bundle){
        bundle.setClassLoader(getClass().getClassLoader());
        Rect rect = bundle.getParcelable("rect");
        process(rect); // Do more with the parcelable.
    }
};

调用IPC方法

如要调用通过 AIDL 定义的远程接口,调用类必须执行以下步骤:

  1. 在项目的 src/ 目录中加入 .aidl 文件。(也可以在任意地方编写aidl文件,通过android sdk提供的aidl命令行工具生成同名的java抽象实现)
  2. 声明一个 IBinder 接口实例(基于 AIDL 生成)。
  3. 实现 ServiceConnection。
  4. 调用 Context.bindService(),从而传入您的 ServiceConnection 实现。
  5. 在 onServiceConnected() 实现中,您将收到一个 IBinder 实例(名为 service)。调用 YourInterfaceName.Stub.asInterface((IBinder)service),以将返回的参数转换为 YourInterface 类型。
  6. <font color=BrulyWood>调用您在接口上定义的方法。您应始终捕获 DeadObjectException 异常,系统会在连接中断时抛出此异常。您还应捕获 SecurityException 异常,当 IPC 方法调用中两个进程的 AIDL 定义发生冲突时,系统会抛出此异常。</font>
  7. 如要断开连接,请使用您的接口实例调用 Context.unbindService()。
上一篇 下一篇

猜你喜欢

热点阅读