跨进程高并发时使用
流程
服务端:创建一个service来监听客户端的链接请求,然后创建一个AIDL文件,暴露给客户端,把想要暴露给客户端的接口在这里声明,最后在service中实现这个AIDL接口
客户端:绑定服务端service,将服务端返回的binder对象转成AIDL接口所属类型,接着就可以调用AIDL中的方法了
AIDL支持的数据类型
1.基本数据类型
2.String和CharSequence
3.List:只支持ArrayList,每个元素必须被AIDL支持
4.Map:只支持hashMap,每个元素必须被AIDL支持(key&vaule)
5.Parcelable:所有实现Parecelable接口的对象
6.AIDL:所有AIDL接口本身也可以在AIDL文件中使用
语法
1.所有使用到的对象必须import进来,不管是不是在同一个包下
2.如果AIDL文件使用到了自定义的Parcelable对象,那么必须新建一个和它同名的AIDl文件,并且声明为Parcelable类型
例如:BookManager.aidl 和 Book.aidl
package com.smt.demo;
parcelable Book;
3.不支持静态常量
4.方法参数必须加上 in,out,inout
in:输入类型参数(在service中改变该对象属性,原始对象不会改变)
out:输出类型参数(在service中改变该对象属性,原始对象会改变)
inout:输入输出类型参数
根据情况使用,out&inout是有开销的
5.AIDL包接口需要一致(客户端&服务端),客户端中需要反序列化服务端中的AIDL接口相关的类,可以把AIDL文件放到同一个包下,方便移植
使用as创建AIDL文件,找不到类问题
解答:在build.gradle 文件 Android 中加入
sourceSets {
main {
java.srcDirs = ['src/main/java', 'src/main/aidl']
}
}
具体语法请参考BookManager.aidl Book.aidl
注意事项
1.在实现services中方法时,需要加同步,因为AIDL方法是在服务端的Binder线程池中执行的,当多个客户端同时连接时会出现并发访问
BookManager.Stub mBookManager = new BookManager.Stub() {
@Override
public List
synchronized (mBooks){
return mBooks;
}
}
};
2.对象不能跨进程传输,本质上是对象的反序列化过程(A 对象传到其他进程时,是重新创建的一个A对象,两个A对象不相等)
观察者模式解绑遇到的问题
可以使用RemoteCallbackList解决,RemoteCallbackList
ArrayMap
的同一个对象会在服务端生成不同的对象,但是这些新对象的Binder对象是同一个,当时客户端
进程挂掉之后会自动帮我们移出Listenr,RemoteCallbackList内部实现了现场同步的功能
使用方式:
RemoteCallbackList
int count = callbackList.beginBroadcast();
for (int i = 0; i < count ; i++) {
BookListener broadcastItem = callbackList.getBroadcastItem(i);
//TODO you code...
}
callbackList.finishBroadcast();
3.尽量不要在UI线程调用服务端的远程方法(可能会导致ANR),onServiceConnected和onServiceDisconnected方法也都运行在UI线程中,避免调用服务端耗时方法
4.由于服务端方法是运行的binder线程池中的,所以可以执行耗时方法,这时轻易不要开现场去执行,当服务端调用客户端的方法时,客户端方法运行在客户端的binder线程池中,所以不能做耗时操作和更新UI操作
5.当Binder意外死亡时,可以选择重连
两种方式
1.DeathRecipient,当Bander死亡时会回调binderDied,可在此方法进行重连操
作
2.onServiceDisconnected中重连操作
两种方式的区别:onServiceDisconnected运行在UI线程,DeathRecipient方式
运行在客户端的Binder线程中
6.权限验证:在onTransact方法中验证,可以使用自定义权限,包名uid,pic等来验证,返回false表示调用服务端方法失败(需要拷贝AIDL生成的java文件进行修改)