第十八周:AIDL使用及其异步操作

一、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
service_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如下:


点击_log

service_log

依旧看时间点,这个很好理解吧。
所以区别就是oneway就是交给binder线程池处理,所以还是自己处理比较灵活。
至于回调就用RemoteCallbackList啦,具体使用就看艺术探索吧,不是今天的主题,溜了溜了。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容