4双通道连接分析
双通道只能是一对一的通信,服务器端需要知道当前连接的是哪个客户端,通信双方需要建立双向连接,不区分客户端和服务端。
4.1 connecte过程
首先回忆一下单通道建立连接的过程,
1,获取服务端的Messenger对象,创建本地的Handler对象, 创建AsyncChannel对象。
2,调用本地AsyncChannel对象的 connect()方法。
3,处理CMD_CHANNEL_HALF_CONNECTED消息。
当然双通道的建立连接过程比单通道要复杂,其实换一个角度想也并不复杂。既然双通道不区分客户端和服务端,那么可以将双通道
看成2个单独单通道,这样理解就很清楚了。
虽然服务端连接客户端和客户端连接服务端完全一样,那么客户端连接服务端之后,服务端什么时候连接客户端呢?答案就在最后一个步骤中。
客户端和服务端连接之后会处理CMD_CHANNEL_HALF_CONNECTED,既然如此,可以在客户端处理该消息时给服务端发送消息,
服务端接收到该消息之后发起连接过程。
客户端处理如下,
case AsyncChannel.CMD_CHANNEL_HALF_CONNECTED: {
mClientAsyncChannel.sendMessage(AsyncChannel.CMD_CHANNEL_FULL_CONNECTION);
break;
}
客户端发送消息过程就不论述了,见3.3小节。
服务端收到消息之后发起连接请求,
case AsyncChannel.CMD_CHANNEL_FULL_CONNECTION: {
mServiceAsyncChannel.connect(AsyncChannelService.this, mServiceHandler, msg.replyTo);
break;
}
注意此时的参数代表的意义:
第一个参数是服务端的上下文,
第二个参数是服务端的Handler对象,
第三个参数是客户端的Messenger对象。
服务端调用AsyncChannel的connect方法就不再论述了,和3.2小结完全一样。
服务端和客户端连接之后,就可以不用刻意区分服务端和客户端了。
双通道连接过程如下,
1,客户端调用AsyncChannel对象的 connect()方法进行连接。
2,连接成功客户端收到CMD_CHANNEL_HALF_CONNECTED消息之后向服务端发送CMD_CHANNEL_FULL_CONNECTION消息。
3,服务端收到CMD_CHANNEL_FULL_CONNECTION消息之后调用AsyncChannel对象的 connect()方法进行连接。
4,连接成功之后功客户端会收到CMD_CHANNEL_HALF_CONNECTED消息。
4.2 fullyConnectSync连接
客户端直接调用AsyncChannel的fullyConnectSync方法进行连接,
int result = mClientAsyncChannel.fullyConnectSync(this, clientHandler, serviceHandler);
if (AsyncChannel.STATUS_SUCCESSFUL == result) {
•••
}
AsyncChannel的fullyConnectSync方法如下,
public int fullyConnectSync(Context srcContext, Handler srcHandler, Handler dstHandler) {
int status = connectSync(srcContext, srcHandler, dstHandler);
if (status == STATUS_SUCCESSFUL) {
Message response = sendMessageSynchronously(CMD_CHANNEL_FULL_CONNECTION);
status = response.arg1;
}
return status;
}
connectSync方法直接调用connected方法进行初始化,
然后调用sendMessageSynchronously发送同步消息CMD_CHANNEL_FULL_CONNECTION,
同步消息的处理流程见3.4,在此就不论述了。
服务端的CMD_CHANNEL_FULL_CONNECTION消息处理如下,
case AsyncChannel.CMD_CHANNEL_FULL_CONNECTION: {
mServiceAsyncChannel.connect(AsyncChannelServiceFull.this, mServiceHandler, msg.replyTo);
mServiceAsyncChannel.replyToMessage(msg, AsyncChannel.CMD_CHANNEL_FULLY_CONNECTED, AsyncChannel.STATUS_SUCCESSFUL);
break;
}
当服务端接收到该请求后,如果同意建立双向通道,在connect()之后给出OK的回应.
这样就完成了连接。
在开发中一般调用AsyncChannel的fullyConnectSync进行连接。