最近在看柯元旦的《Android内核剖析》,做一些小结。
-
Binder是一种架构,其中有三个接口
- 客户端接口
- 服务端接口
- 驱动端接口
客户端和服务端分属两个不同进程,所以通过驱动端来进行进程间通信
-
Binder架构
从柯元旦的《Android内核剖析》中截图
-
服务端一被创建,内部就启动一个隐藏线程不断接受来自驱动端的消息,从消息中取出参数传入并调用
onTransact()
方法,在内部根据onTransact()
方法的具体方法实现,所以服务端需要重载onTransact()
方法 。
onTransact(int code,Parcel data,Parcel reply,int flag)
方法的重载,根据int code
来判断调用服务端的哪个函数方法,通过传来的Parcel data
获取所调用函数方法所需的的参数,Parcel reply
则是一个传递结果的变量,可以将被调用方法返回的结果添加在里面,使客户端可以使用,int flag
是执行IPC调用的模式,常量 0 表示双向,说明需要服务端需要返回一定的数据,常量 1 表示单向,不需要返回数据。
- 驱动端,任何一个服务端被创建后,同时在驱动端也会创建一个Binder对象mRemote,而客户端则是通过这个mRemote来访问远程服务。
- 客户端
- 如何获得驱动端mRemote
在客户端中,有两个方法可以和服务端建立连接:startService(Intent intent)
/bindService(Intent intent,ServiceConnection conn,int flag)
第一个方法仅仅是开启一个服务端,并没有获得mRemote的引用
第二个方法中的第二个参数ServiceConnection conn
则提供了获得mRemote引用的机会。
public void ServiceConnection{
public void onServiceConnected(ComponentName name,IBinder service);
public void onServiceDisconnected(ComponentName name);
}
可见ServiceConnection
是一个接口,其中的onServiceConnected(ComponentName name,IBinder service)
方法的第二个参数IBinder service
就是mRemote引用,可以实现这个接口的方法时将该IBinder service
保存成类成员变量,方便使用。
如何使用mRemote
根据服务端定义的具体实现方法,在调用mRemote.transact(int code,Pracel data,Pracel reply,int flag)
方法前,先将data包装好,这里要求包装数据的顺序跟在服务端定义的onTransact()
方法中从data取出数据的顺序要一致;如果有数据的话就从reply取出,这里同样要注意顺序。知识拓展
-
aidl工具
该工具是由android SDK提供的,让程序员不用在意于包裹内参数获取或者存放顺序,aidl工具将一个aidl文件转化成java文件,这个java文件中完成了以下几个任务:- 定义一个Java interface ,内部包含aidl文件声明的服务函数,该类基于IInterface接口,即需要提供一个
asBinder()
函数 - 定义一个Proxy类,该类作为客户端程序访问服务端的代理。所谓代理就是为了统一包裹内参数获取或者存放顺序。
- 定义一个Stub类,这是一个abstract类,基于Binder类,实现了java interface。
- 定义一个Java interface ,内部包含aidl文件声明的服务函数,该类基于IInterface接口,即需要提供一个