一、Android中AIDL的作用是什么?
用于Android跨进程通信
二、它所支持的参数类型是?
1、所有的Java基本数据类型,比如int,long,boolean,char,float等。
2、String
3、CharSequence
4、List,List当中的数据元素必须能够被AIDL支持
5、Map:只支持HashMap,里面的每个元素都必须被AIDL支持,包括key和value
6、Parcelable:实现了Parcelable接口的对象
7、AIDL接口:所有的AIDL接口本身也可以在AIDL文件中使用。
三、默认情况下AIDL的调用过程是同步还是异步?如何指定AIDL为异步调用?
默认情况时同步调用。所以客户端调用服务端的耗时操作时会报ANR,反之同理。
1、自行实现多线程
当服务端(客户端)需要做耗时操作时,手动开启线程或利用线程池等做异步操作。
2、利用oneway关键字实现
当使用oneway的时候,远程调用不会阻塞,会立即返回。
与手动实现的区别就是oneway调用远程后立刻返回,不会管远程具体实现。而前者会进入远程方法。下面我们来写一个demo理解。
为了简化代码,activity代码就贴一下点击事件,service也只贴binder部分了,由于kotlin老是忘,所以现在写demo一般都用kotlin,代码比较简单相信大家能看明白
activity:
override fun onClick(v: View?) {
when (v?.id) {
R.id.get_connect -> {
val mIntent = Intent(this, AService::class.java)
bindService(mIntent, connection, Context.BIND_AUTO_CREATE)
}
R.id.do_thing -> {
Log.e("test", "client click")
binder?.longTimeAction()
}
}
}
service:
val test = object : ITestInterface.Stub() {
override fun longTimeAction() {
Log.e("test", "threadName:" + Thread.currentThread().name)
var j = i
i++
Log.e("test", "sleep_start" + j)
Thread.sleep(10000)
Log.e("test", "sleep_finish" + j)
}
};
AIDL:
interface ITestInterface {
oneway void longTimeAction();
}
首先我们尝试oneway方式异步,我们连续点击3次按钮IPC通信,log如下:
观察log时间点,可以看出点击是在同一时间发生的,而服务端是在binder_1这个binder线程池里单线程执行的,所以达到了异步需求,注意这里并不是主线程哦,艺术探索里也提到了。
接下来稍微修改一下service 的代码,把他换到自己的线程池里执行:
lateinit var execute : ExecutorService
val test = object : ITestInterface.Stub() {
override fun longTimeAction() {
execute.execute{
Log.e("test", "threadName:" + Thread.currentThread().name)
var j = i
i++
Log.e("test", "sleep_start" + j)
Thread.sleep(10000)
Log.e("test", "sleep_finish" + j)
}
}
};
var i: Int = 0
override fun onCreate() {
super.onCreate()
execute = Executors.newFixedThreadPool(5, Executors.defaultThreadFactory())
}
log如下:
依旧看时间点,这个很好理解吧。
所以区别就是oneway就是交给binder线程池处理,所以还是自己处理比较灵活。
至于回调就用RemoteCallbackList啦,具体使用就看艺术探索吧,不是今天的主题,溜了溜了。