远程对象的基础接口,是一个为了在执行进程中和进程间调用时的高性能,而设计的轻量级远程调用机制的核心部分。这个接口描述了和远程对象交互的抽象协议。不要直接实现这个接口,而是通过继承Binder来实现。
IBinder的关键API是与 Binder.onTransact()相匹配的transact().这个方法分别允许你给IBinder对象发出一个请求,并接收一个进入一个Binder对象的请求。这个事务API是同步的,这样一个对transact()的调用会在目标从Binder.onTransact()返回后自己才能返回;这是在调用一个存在于本地进程对象的预期行为,并且潜在的IPC(进程间通信机制)也在进程间切换时确保这些相同语义得到应用。
通过transact()送出的数据是一个 Parcel,一个通常的数据缓冲器,并且包含一些关于他内容的元数据。这些元数据用来在缓冲期内管理IBinder对象的引用,以至于那些引用可以在缓冲器穿越进程时被包含进来。这个机制可以确保了当一个IBinder被写入一个Parcel被传送给其他进程时,如果其他进程返回给原先进程同样的IBinder引用,原先的进程接收同样的IBinder对象。这些语义允许IBinder/Binder对象被用作进程间切换时管理的唯一的身份(作为一个标志或其他目的来服务)。
系统在每一个事务线程运行的进程里维护了一个事务线程池。这些线程用来分发所有来自其他进程的IPCs。例如,当一个IPC从A进程到B进程被建立,在A里的调用线程用transact()向进程B发出一个事务。在B中下一个有效的线程池接收到进入的事务,对目标对象调用Binder.onTransact(),并且用结果Parcel回复。在接收到结果时,在进程A中的线程返回以允许进程继续执行。效果上,其他进程看起来象作为那些你没有在自己进程中创建执行的额外线程而使用的。
Binder系统也支持进程间的递归调用。例如,如果进程A执行一个到进程B的事务,并且进程B在处理这个事务时对一个在A里实现的IBinder调用transact(),然后在A里的正在等待原先事务结束的线程,将会关心对被B调用的对象调用Binder.onTransact().这个就确保在调用远程binder对象时的递归语义和调用本地对象时是一样的。
当和远程对象打交道时,你经常想要找出何时他们不再有效。下面有三条可以被确定的方式:
- 当你试图对一个所属进程已经不存在的IBinder调用transact()方法时,会抛出一个RemoteException异常。
- pingBinder()可以被调用,并且如果远程进程不再存在会返回假。
- linkToDeath()方法可以用来用这个IBinder注册一个IBinder.DeathRecipient ,当他包含的进程消失的时候被调用。
IBinder.DeathRecipient | 当服务一个IBinder的进程消失后接收一个回调的接口。 |
请参阅
概要
常量
Value | ||||
---|---|---|---|---|
int | DUMP_TRANSACTION | IBinder事务协议码:清除内部状态 | 1598311760 | 0x5f444d50 |
int | FIRST_CALL_TRANSACTION | 用户指令的第一个事务码可用。 | 1 | 0x00000001 |
int | FLAG_ONEWAY | transact(int,Parcel, Parcel, int)标志符: 这是一个单向调用,意味着调用者会立即返回,而不等待从被调用者那里的结果。 | 1 | 0x00000001 |
int | INTERFACE_TRANSACTION | IBinder事务协议码:向事务接收端询问他的规范接口描述符。 | 1598968902 | 0x5f4e5446 |
int | LAST_CALL_TRANSACTION | 用户指令的最后一个事务码可用。 | 16777215 | 0x00ffffff |
int | PING_TRANSACTION | IBinder事务协议码:pingBinder(). | 1599098439 | 0x5f504e47 |
公共方法
String | getInterfaceDescriptor() | |||||
获得被这个binder支持的接口的规范名字。 | ||||||
boolean | isBinderAlive() | |||||
检查binder所在的进程是否还是存在的。 | ||||||
void | linkToDeath(IBinder.DeathRecipientrecipient, int flags) | |||||
如果这个binder消失,为标志信息注册一个接收器。 | ||||||
boolean | pingBinder() | |||||
检查是否这个对象还存在。 | ||||||
IInterface | queryLocalInterface(Stringdescriptor) | |||||
试图获得一个对这个Binder对象的一个接口的本地实现。 | ||||||
boolean | transact(intcode,Parceldata, Parcelreply, int flags) | |||||
用对象执行一个一般的操作。 | ||||||
boolean | unlinkToDeath(IBinder.DeathRecipientrecipient, int flags) | |||||
清除一个之前注册的死亡标识信息。 |
内容
常量
public static final intDUMP_TRANSACTION
public static final intFIRST_CALL_TRANSACTION
public static final intFLAG_ONEWAY
public static final intINTERFACE_TRANSACTION
public static final intLAST_CALL_TRANSACTION
public static final intPING_TRANSACTION
公共方法
public StringgetInterfaceDescriptor()
public booleanisBinderAlive()
返回值
- 如果进程不能在则返回假值。注意如果返回真值,进程可能在调用返回时已经死掉了。
public void linkToDeath(IBinder.DeathRecipientrecipient, int flags)
你将仅仅从远程binders那里接收到一个死亡标识信息,作为定义中的本地的binders在调用死亡前是不会死亡的。
异常
RemoteException如果目标IBinder的进程已经死亡。 | |
RemoteException |
public boolean pingBinder()
返回值
- 如果服务进程已经消失则返回假值,否则结果被pingBinder()在另一边实现返回(通常默认的是真值)。
public IInterfacequeryLocalInterface(Stringdescriptor)
public boolean transact(intcode, Parceldata, Parcelreply, int flags)
参数
code | 执行的行为. 这将会是一个在 FIRST_CALL_TRANSACTION和 LAST_CALL_TRANSACTION之间的数字。 |
---|---|
data | 发送给目标的编组后的数据,通常不是null。如果你不传送任何数据,你必须在创建一个空的Parcel放在这里。 |
reply | 来自目标被接收的编组后的数据。如果你对返回值不感兴趣可能返回null。 |
flags | 额外的操作符。0是通常意义的RPC,或者 FLAG_ONEWAY表示一个单向RPC. |
public boolean unlinkToDeath(IBinder.DeathRecipientrecipient, int flags)
清除一个之前注册的死亡标识信息。如果这个对象已经死亡,这个接收器将不再被调用。
返回值
- 如果接收器成功被断开连接则返回真,你必须确保他的 DeathRecipient.binderDied()方法没有被调用。如果目标IBinder已经死亡则返回假值,意味着这个方法已经(或将要)被调用。
异常
NoSuchElementException如果给定的接收器还没有和IBinder注册,并且这个IBinder还活着。注意如果接收器从来没有被注册过,但是IBinder已经死亡,这个异常将不会抛出,你会接收到一个假值作为返回值。 |