前言
AIDL是什么?还记得好多面试知识点中提到,应用之间的跨进程通信有哪些方式,AIDL好几次被作为正确答案来说,其实这是大错特错,其实AIDL就是帮程序员偷懒的封装类。AIDL只是对Binder和BinderProxy对象进行一层分装。
Binder和BinderProxy对象的理解
进程A在自己程序中new一个Binder对象,进程B可以通过ServiceManager拿到Binder对象的客户端代理类BinderProxy对象。当进程B通过调用BinderProxy的对象来与进程A中Binder类进行跨进程通信,具体怎么调用的请看下图:
注意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