AsyncChannel类源码简要分析
概述
这个类提供了两个Handler之间进行异步通信的机制。这两个Handler可能位于同一进程中,也可能位于不同的进程中。目前,ASyncChannel可以使用两种协议方式:
一对多的通信方式(简单的请求与应答方式)
在此种交互方式下,服务器端并不知道请求来自哪个客户端, 通信前双方不需要显示建立连接。客户端(发送方)将请求发送给服务器(接收方),服务器则通过replayToMessage方法向客户端发送应答消息。主要的交互序列图如下所示:一对一的通信方式
在此种通信模式下,服务器端需要知道当前连接的是哪个客户端,通信双方需要显示建立双向连接。主要的交互序列图如下所示:
主要方法分析
- connected
public void connected(Context srcContext, Handler srcHandler, Messenger dstMessenger) {
if (DBG) log("connected srcHandler to the dstMessenger E");
// Initialize source fields
mSrcContext = srcContext;
mSrcHandler = srcHandler;
mSrcMessenger = new Messenger(mSrcHandler);
// Initialize destination fields
mDstMessenger = dstMessenger;
if (DBG) log("connected srcHandler to the dstMessenger X");
}
ASyncChannel提供了几个connect系列的函数,主要都是围绕上面这个函数进程的。它将两个重要的成员变量mSrcMessenger和mDstMessenger的值确定下来,为接下来的消息发送和接收做准备。
- sendMessage
public void sendMessage(Message msg) {
msg.replyTo = mSrcMessenger;
try {
mDstMessenger.send(msg);
} catch (RemoteException e) {
replyDisconnected(STATUS_SEND_UNSUCCESSFUL);
}
}
同样ASyncChannel也定义许多sendMessage接口,不过也是围绕上述函数进行的。该函数通过mDstMessenger将消息发送给接收方(服务器端)。
- replyToMessage
public void replyToMessage(Message srcMsg, Message dstMsg) {
try {
dstMsg.replyTo = mSrcMessenger;
srcMsg.replyTo.send(dstMsg);
} catch (RemoteException e) {
log("TODO: handle replyToMessage RemoteException" + e);
e.printStackTrace();
}
}
ASyncChannel也定义了许多replyToMessage接口,不过也是围绕上述函数进行的。它将服务器的响应消息发送给之前消息的发送方。
此外,ASyncChannel也定义同步发送消息的方法sendMessageSynchronously
。
结语
在WiFi相关的代码中,定义了许多状态机,其中大量使用了ASyncChannel提供的同步和异步方法,来实现状态机内部以及状态机之间的消息通信。而ASyncChannel本身的实现机制其实依赖于Android Handler类,它本身提供了线程间或进程间的消息通信,具体可参考之前对Handler类的分析文章:
1. 基于Message的线程间通信实例
2. 基于Message的进程间通信实例