AIDL 从入门到精通

什么是 RPC 框架:- 很简单,就是用来实现远程调用,进程间通信的,不同的系统由不同的实现,android 这里的实现叫 Binder

不同的系统有不同的实现,但是原理基本都相同,理解起来很简单:

即将调用的方法(接口名、方法名、参数类型、参数)序列化之后发送到远端,在远端反序列化之后调用接口的实现类的方法(接口主要是为了使用动态代理),2 端的实现类通过内部的通信实现诸如发射参数,实现等待,接受返回值等远程跨进程操作。所以我们在实现RPC框架的时候需要选择合适的序列化与反序列化方式


AIDL 是 Android 特有的 IPC 进程间通讯方式

AIDL 的写法其实和绑定服务的代码差不多,IBander 也是 android 默认提供的一个 AIDL 接口

需要注意的是 5.0 之后,不能隐式启动 service,不能想以前一样定义 action 来启动服务了,尤其是不是跨应用启动服务,这也算是一种安全上的考虑

若是想非常详细的了解 AIDL ,请看慕课网的科普视频

AIDL 写法

  1. 使用 Android 提供的方式生命一个 AIDL 接口


    Snip20171024_3.png

然后在这个AIDL 接口中声明业务需要的方法

// IBanZheng.aidl
package com.bloodcrown.bcremoteservice.aldl;

// Declare any non-default types here with import statements

interface IBanZheng {

    void banZheng();

    /**
     * Demonstrates some basic types that you can use as parameters
     * and return values in AIDL.
     */
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat,
            double aDouble, String aString);
}

这就是系统帮我们创建的 AIDL 接口,在里面我写了一个 banZheng() 的方法,剩下的都是系统的事了,Android 系统会根据我们声明的这个 AIDL 接口创建一个相关的 IPC 通讯类出来:

位置在:


Snip20171024_4.png

系统会在帮我们创建出一个单独的 aidl 包出来,里面放我们声明的 AIDL 类,注意不是实现类


Snip20171024_6.png

详细的代码,有点长,系统帮我们添加的代码都是进行 ipc通讯的

package com.bloodcrown.bcremoteservice.aldl;
// Declare any non-default types here with import statements

public interface IBanZheng extends android.os.IInterface {
    /**
     * Local-side IPC implementation stub class.
     */
    public static abstract class Stub extends android.os.Binder implements com.bloodcrown.bcremoteservice.aldl.IBanZheng {
        private static final java.lang.String DESCRIPTOR = "com.bloodcrown.bcremoteservice.aldl.IBanZheng";

        /**
         * Construct the stub at attach it to the interface.
         */
        public Stub() {
            this.attachInterface(this, DESCRIPTOR);
        }

        /**
         * Cast an IBinder object into an com.bloodcrown.bcremoteservice.aldl.IBanZheng interface,
         * generating a proxy if needed.
         */
        public static com.bloodcrown.bcremoteservice.aldl.IBanZheng asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
            if (((iin != null) && (iin instanceof com.bloodcrown.bcremoteservice.aldl.IBanZheng))) {
                return ((com.bloodcrown.bcremoteservice.aldl.IBanZheng) iin);
            }
            return new com.bloodcrown.bcremoteservice.aldl.IBanZheng.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_banZheng: {
                    data.enforceInterface(DESCRIPTOR);
                    this.banZheng();
                    reply.writeNoException();
                    return true;
                }
                case TRANSACTION_basicTypes: {
                    data.enforceInterface(DESCRIPTOR);
                    int _arg0;
                    _arg0 = data.readInt();
                    long _arg1;
                    _arg1 = data.readLong();
                    boolean _arg2;
                    _arg2 = (0 != data.readInt());
                    float _arg3;
                    _arg3 = data.readFloat();
                    double _arg4;
                    _arg4 = data.readDouble();
                    java.lang.String _arg5;
                    _arg5 = data.readString();
                    this.basicTypes(_arg0, _arg1, _arg2, _arg3, _arg4, _arg5);
                    reply.writeNoException();
                    return true;
                }
            }
            return super.onTransact(code, data, reply, flags);
        }

        private static class Proxy implements com.bloodcrown.bcremoteservice.aldl.IBanZheng {
            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 void banZheng() throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    mRemote.transact(Stub.TRANSACTION_banZheng, _data, _reply, 0);
                    _reply.readException();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
            }

            /**
             * Demonstrates some basic types that you can use as parameters
             * and return values in AIDL.
             */
            @Override
            public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                    _data.writeInterfaceToken(DESCRIPTOR);
                    _data.writeInt(anInt);
                    _data.writeLong(aLong);
                    _data.writeInt(((aBoolean) ? (1) : (0)));
                    _data.writeFloat(aFloat);
                    _data.writeDouble(aDouble);
                    _data.writeString(aString);
                    mRemote.transact(Stub.TRANSACTION_basicTypes, _data, _reply, 0);
                    _reply.readException();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
            }
        }

        static final int TRANSACTION_banZheng = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
        static final int TRANSACTION_basicTypes = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
    }

    public void banZheng() throws android.os.RemoteException;

    /**
     * Demonstrates some basic types that you can use as parameters
     * and return values in AIDL.
     */
    public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, java.lang.String aString) throws android.os.RemoteException;

其实我们主要看这个类:

 public static abstract class Stub extends android.os.Binder implements com.bloodcrown.bcremoteservice.aldl.IBanZheng

和我们在绑定服务时,在服务中写的用户返回的内部类一样不一样,都是继承 Binder 类,实现我们自己声明的接口,然后我们使用也是使用这个stub 类,我们的目的就是让系统帮我们创建出这个 stub 类

  1. 在 service 中使用这个 stub 类

现在我们写的内部类直接继承这个 stub 即可

 class MediaIBander extends IBanZheng.Stub {

        @Override
        public void banZheng() throws RemoteException {
            Log.d(TAG, "办证啦...");
        }

        @Override
        public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {

        }
    }

然后在绑定生命周期函数内返回这个内部类

@Override
    public IBinder onBind(Intent intent) {
        Log.d(TAG, "remoteService - onBind...");
        return new MediaIBander();
    }
  1. 在 activity 中接受数据,转换对象类型
 public class MyRemoteServiceConnection implements ServiceConnection {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iBanZheng = IBanZheng.Stub.asInterface(service);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    }

AIDL 深入理解

AIDL 是 android 系统 IPC 进程间通讯协议,但是记住仅仅只是 android ,换个平台就不是 AIDL 了。

android 中每个进程都有自己独立的虚拟机 JVM , 每个JVM 的内存时独立的,所以进程间通信依靠传递对象引用是不行的,因为内存是不连续的, A进程 的内存中有的对象,你把对象引用给到 B进程内存里面可是没这个对象的,所以 google 就提供了 AIDL

AIDL 是一个桥


进程1 的请求会通过 AIDL 发送给系统,系统根据请求标识找到进程2,把进程1 的请求交给进程2去处理,同理进程2处理完后把结果通过 AIDL 再发送给进程1

AIDL 只支持基本数据类型,集合,Parcelable 序列化类型数据的传输


AIDL 方法种若要传递对象类型,对象类型需要实现 Parcelable 序列化接口。Parcelable 的原理就是把大的数据类型对象,打散成一个个系统能支持的基本数据类型数据,然后集中打个包传递过去,到目标进程再组合成对象类型。这个过程也叫打包,拆包


系统帮我们生成的 AIDL 对象,里面一个 Stub 类型的内部类,Stub 里面又有一个 Proxy 类型的内部类

IPC 通信的核心方法就在于期中的 transact 和 onTransact 方法

  • transact 会调用系统底层 IPC 去传递数据
  • onTransact 会接受系统底层 IPC 传递过来的数据

我们跟着代码来看一下:

  1. 声明一个 AIDL 类,内部有一个叫 banZheng 的方法
interface IBanZheng {

    void banZheng();
}
  1. 获取远程进程代理
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iBanZheng = IBanZheng.Stub.asInterface(service);
        }

上面这是客服端获取远程服务的 binder ,我们跟进去看看

        public static com.bloodcrown.bcremoteservice.aldl.IBanZheng asInterface(android.os.IBinder obj) {
            if ((obj == null)) {
                return null;
            }
            android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
            if (((iin != null) && (iin instanceof com.bloodcrown.bcremoteservice.aldl.IBanZheng))) {
                // 是同一个进程,返回 Stub 对象本身
                return ((com.bloodcrown.bcremoteservice.aldl.IBanZheng) iin);
            }
            // 不是同一个进程,返回远程进程的代理
            return new com.bloodcrown.bcremoteservice.aldl.IBanZheng.Stub.Proxy(obj);
        }

这个方法是 Stub 的方法,我们可以看到,当服务端和客户端不再同一个进程时,其实我们拿到的只是远程进程的代理类,这个代理类会帮我们进程 IPC 底层的通讯

  1. Proxy 调用底层通讯方法
            @Override
            public void banZheng() throws android.os.RemoteException {
                android.os.Parcel _data = android.os.Parcel.obtain();
                android.os.Parcel _reply = android.os.Parcel.obtain();
                try {
                    // 先把数据打包成可以通过 IPC 通讯传递
                    _data.writeInterfaceToken(DESCRIPTOR);
                    // 调用底层 IPC 方法传递数据(此时线程会挂起,也就是卡线程了)
                    mRemote.transact(Stub.TRANSACTION_banZheng, _data, _reply, 0);
                    // 等待远程返回结果(此时线程会激活,重新执行下面任务)
                    _reply.readException();
                } finally {
                    _reply.recycle();
                    _data.recycle();
                }
            }

Proxy 实现了我们声明的 AIDL 接口,所以我们掉远程 binder 的方法时就是调用的 Proxy 的相关方法,我们看上面的方法,先把数据序列化打包,再调用底层的 transact 方法进行 IPC 通讯,然后等待远程返回结果,这里会卡住线程,所以客户端的 IPC 请求最好在非 UI 线程执行

  1. Stub 的 onTransact 方法是服务端核心,服务端在 onTransact 方法种接受到客服端传过来的参数,然后计算,再把结果写会去,客户端才能收到结果,注意服务端的 binder 是在系统的 binder 线程池中执行的,不要我们自己再起线程了。

AIDL 远程是异步方法

不知道大家有没有想过,远端若是耗时方法的话,会不会卡客户端线程,我们来测试一下,在两端分别打印日志,标记关键节点时间,远端延迟3秒返回数据,大家看一下结果就能清楚了

客户端
服务端

看日志很清楚了吧,客户端调 AIDL 方法可是会卡主线程的, 所以我们需要注意啊,远端若是耗时的话,我们需要在客户端点开线程,再进行 AIDL 远程通信

我们现在知道了客户端的 AIDL 方法是异步任务会卡主线程,那么大家就不想知道服务端的方法是怎么运行的吗

系统有句话这么说:

服务端的 binder 方法运行在系统的 AIDL 线程池里

换句话说,AIDL 在服务端已经跑在单独的线程里了,不用我们自己开线程了,这里我测试了下,打印服务端启动时所在线程和 binder 执行任务时所在线程

看日志就清楚了把,系统对于 AIDL 在服务端是有优化的,自动开线程池


AIDL 双向通讯

AIDL 是单项通讯的,假设我们实现了 AIDL A -> B 进程的通信,那么在 A binder 联通的时刻把 A 实现的 AIDL.Stub 传递给 B ,B 就能通过这个传过来的 AIDL.Stub 实现 B -> A 的通讯了,AIDL.Stub 对象可以直接 IPC 传递的

参考例子可以看:


binder 如何理解

先看图:


binder 和 AIDL 一样都是 android 的 IPC 通信机制,不同于 unix 其他的 IPC ,binder 对每个进程的 UID 支持非常好,安全性高

至于我们在使用时感觉 binder 不是夸进程的,那是错觉。android 4大组件都是有声明周期的,什么时候 应该怎么运行都是由 ActivityManageService 控制的,看上面的图应该知道 binder 是组件与 ActivityManageService 通信的,若 Activity 与 Service 在同一个进程,那么内存共享,传递的数据可以给过去,这就是我们常见的与 Service 的通信。

若 Activity 与 Service 不在同一个进程,内存不能共享,那么数据就得通过 android 系统特有的 AIDL IPC 通道先序列化然后过去再反序列化,AIDL 的意思就在于跨进程传递数据了

详细的大家请看:


AIDL 解读补充

AIDL 跨进程通信的核心 Proxy ,大家看构造方法,注意 remote 就是那个远程 Service,remote 恰恰就是 IBinder 对象,所以 IBinder 才是 android 中 IPC 通讯的基石

private static class Proxy implements com.lypeer.ipcclient.BookManager {
    private android.os.IBinder mRemote;

    Proxy(android.os.IBinder remote) {
        //此处的 remote 正是前面我们提到的 IBinder service
        mRemote = remote;
    }
 
    @Override
    public java.util.List<com.lypeer.ipcclient.Book> getBooks() throws android.os.RemoteException {
        //省略
    }

    @Override
    public void addBook(com.lypeer.ipcclient.Book book) throws android.os.RemoteException {
        //省略
    }
    //省略部分方法
}

我们在客户端抓换 binder 为指定接口时系统的操作,判断 binder 在本进程有没有实例,就是客户端和服务端是不是在一个进程,是的话就返回对象,queryLocalInterface 方法就是干的这事,不在一个进程的话,就需要 proxy 代理了,proxy 代理了 AIDL 实现的接口方法里的数据序列化,反序列化,至于 IPC 通信靠的还是 binder 去实现

public static com.lypeer.ipcclient.BookManager asInterface(android.os.IBinder obj) {
    //验空
    if ((obj == null)) {
        return null;
    }
    //DESCRIPTOR = "com.lypeer.ipcclient.BookManager",搜索本地是否已經
    //有可用的对象了,如果有就将其返回
    android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
    if (((iin != null) && (iin instanceof com.lypeer.ipcclient.BookManager))) {
        return ((com.lypeer.ipcclient.BookManager) iin);
    }
    //如果本地没有的话就新建一个返回
    return new com.lypeer.ipcclient.BookManager.Stub.Proxy(obj);
}

我们在客户端调用 AIDL 的方法,过程是下面这样走的,核心就是调起 IBinder 类型的 mRemote 对象的 transact 方法,transact 方法就会进行跨进程通信了

@Override
public java.util.List<com.lypeer.ipcclient.Book> getBooks() throws android.os.RemoteException {
    //很容易可以分析出来,_data用来存储流向服务端的数据流,
    //_reply用来存储服务端流回客户端的数据流
    android.os.Parcel _data = android.os.Parcel.obtain();
    android.os.Parcel _reply = android.os.Parcel.obtain();
    java.util.List<com.lypeer.ipcclient.Book> _result;
    try {
        _data.writeInterfaceToken(DESCRIPTOR);
        //调用 transact() 方法将方法id和两个 Parcel 容器传过去
        mRemote.transact(Stub.TRANSACTION_getBooks, _data, _reply, 0);
        _reply.readException();
        //从_reply中取出服务端执行方法的结果
        _result = _reply.createTypedArrayList(com.lypeer.ipcclient.Book.CREATOR);
    } finally {
        _reply.recycle();
        _data.recycle();
    }
    //将结果返回
    return _result;
}

然后在服务端的 onTransact 方法接受远程数据,反序列化出来,执行操作过程,然后通过 reply.writeString 把结果写回去

@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_getBooks: {
            //省略
            return true;
        }
        case TRANSACTION_addBook: {
            //省略
            return true;
        }
    }
    return super.onTransact(code, data, reply, flags);
}

参考自:


使用 Messenger,handle 实现 IPC

Messenger 默认实现了 ibinder ,是对 AIDL 的封装,大体的使用过程如下,我就不贴代码了,看代码的请看:

  1. 服务端实现一个Handler,在 onBind 时 return mMessenger.getBinder() 返回给自客户端用来通信
  2. 客户端接受 messager 对象 Messenger mService = new Messenger(service);
  3. 通过 Messenger 发送消息 mService.send(msg);
  4. 客户端也给服务端提供一个 Messenger 就能实现双向通信了

AIDL中的 in,out,inout

什么是 in,out,inout ,是 AIDL 声明参数在进程2端作用域的标记,看下面这个 AIDL 接口

// BookManager.aidl
package com.lypeer.ipcclient;
import com.lypeer.ipcclient.Book;

interface BookManager {    
    //保证客户端与服务端是连接上的且数据传输正常
    List<Book> getBooks();

    //通过三种定位tag做对比试验,观察输出的结果
    Book addBookIn(in Book book);
    Book addBookOut(out Book book);
    Book addBookInout(inout Book book);
}

AIDL 的参数默认是 in 的,一般我们也不写

  • in : 参数在服务端的任何变化不会映射到客户端
  • out:服务端接受的是一个 空内容的变量对象,在 服务端 对这个变量的任何修改都会映射到客户端
  • inout: 服务端接受的是客户端传过来的参数有内容,在 服务端 对这个变量的任何修改都会映射到客户端,而客户端的修改不会映射到服务端

详细可以看:

最后我说一下,用 AIDL 双向通讯的话不用 in,out 来实现,连官方也不是用 in,out 来实现的,其次我没发现 in,out 的应用场景,暂时作为知识点了解


参考资料

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容