[003]AIDL是什么

前言

AIDL是什么?还记得好多面试知识点中提到,应用之间的跨进程通信有哪些方式,AIDL好几次被作为正确答案来说,其实这是大错特错,其实AIDL就是帮程序员偷懒的封装类。AIDL只是对Binder和BinderProxy对象进行一层分装。

Binder和BinderProxy对象的理解

进程A在自己程序中new一个Binder对象,进程B可以通过ServiceManager拿到Binder对象的客户端代理类BinderProxy对象。当进程B通过调用BinderProxy的对象来与进程A中Binder类进行跨进程通信,具体怎么调用的请看下图:


屏幕快照 2018-11-14 下午11.34.04.png

注意transact方法和onTransact方法的参数

code:int型,这个代表这次通信的requestID,这样子在Binder端就可以根据requestID调用对应的代码了,可以理解为方法名
data:可以理解为方法中的参数
reply:方法的返回值
flag:这个flag代表Binder通信是否是同步还是异步,暂时可以忽略。我们重点关注三个。

假如没有AIDL,一次跨进程的调用的方式是怎么样子:

我用伪代码来实现一下,我们假设进程A把一个整数2传递给进程B,进程B将这个整数的平方4,传递给进程A。

进程A中客户端代码
int getX2result(int x){
  int requestID = 1;
  BinderProxy mClient = ServiceManager.getService("进程B的服务A")
  Parcel data = new Parcel();
  data.writeInt(x)
  Parcel reply = new Parcel():
  mClient.transact(requestID, data,reply,0)//flag设置成0,调用这个方法会跨进程调用Binder服务类中onTransact方法
  int result = replay.readInt();
  return result
}
进程B中服务端代码
protected boolean onTransact(int code, @NonNull Parcel data, @Nullable Parcel reply,
            int flags) throws RemoteException {
  if(code == 1) {
     int x = data.readInt();
     reply.writeInt(getX2result(x));
     return true;
  }
}

int getX2result(int x){
   return x*x;
}

写到这里小编已经累死了,假如一个服务需要一百个这样子的方法,我不得累死,所以科技的进步的本质就是人类偷懒。有人就发现了,上面的伪代码好像都很有规律,能否只提供int getX2result(int x),你就帮我自动生成上面的客户端和服务端的代码。其实AIDL就是这个偷懒的工具。如果大家去看看AIDL生成java类,里面做的事情就是这样子。

在native层中使用BBinder和BpBinder,有没有类似AIDL的工具

可惜没有只能手动去写了

更新:其实AIDL可以生成BBinder和BpBinder,源码下编译用Android.bp就可以了。

小结

所以下次面试,别再说AIDL是跨进程通信的方式,本质是对Binder机制的封装,谢谢,小编要睡觉了。正确android中跨进程通信的方式主要有以下三类:
1.socket通信,应用A初始化请求Zygote孵化出新的进程
2.匿名共享内存,因为Binder不支持图像这样子的大数据传输,所以用匿名共享内存传输应用界面的图像到SurfaceFlinger
3.Binder机制,android系统中到处都是Binder,可以说无Binder无android

彩蛋

手写AIDL实现[027]十分钟让你明白AIDL

你可能感兴趣的:([003]AIDL是什么)