继续话题——软件中的异步非阻塞通讯方式。
由于软件基于MFC开发,所以实现异步通讯时使用了CAsyncSocket类。
首先要了解CAsyncSocket异步机制,引用自
http://blog.csdn.net/tianhai110/article/details/2115270。
由于CAsyncSocket采用的是异步非阻塞机制,所以你随时可以发包,也随时可能收到包。
发送、接收函数都是异步非阻塞的,顷刻就能完成,所以收发交错进行着。也正因为如此,仅调用
它们并不能保障发送或接收的完成。例如发送函数Send,调用它可能有3种结果:错误、部分完成、
全部完成。其中错误又分两种情况:一种是由各种网络问题导致的失败,你需要马上决定是放弃本次操作,
还是启用某种对策;另一种是“忙”,你实际上不用马上理睬。你需要调用GetLastError来判断是哪种情况,
GetLastError返回WSAEWOULDBLOCK,代表“忙”,为什么当你Send得到WSAEWOULDBLOCK却
不用理睬呢?因为CAsyncSocket会记得你的SendWSAEWOULDBLOCK了,待发送的数据会写入
CAsyncSocket内部的发送缓冲区,并会在不忙的时候自动调用OnSend,发送内部缓冲区里的数据。
同样,如果Send只完成了一部分,你也不需要理睬,尚未发送的数据同样会写入CAsyncSocket内部的
发送缓冲区,并在不“忙”的时候自动调用OnSend完成发送。
与OnSend协助Send完成工作一样,OnRecieve、OnConnect、OnAccept也会分别协助Recieve、
Connect、Accept完成工作。这一切都通过消息机制完成。
在你使用CAsyncSocket之前,必须调用AfxSocketInit初始化WinSock环境,而AfxSocketInit会
创建一个隐藏的CSocketWnd对象,由于这个对象由Cwnd派生,因此它能够接收Windows消息。一方面它
会接受各个CAsyncSocket的状态报告,另一方面它能捕捉系统发出的各种SOCKET事件。所以它能够成为
高层CAsyncSocket对象与WinSock底层之间的桥梁:例如某CAsyncSocket在Send时WSAEWOULDBLOCK了,
它就会发送一条消息给CSocketWnd作为报告,CSocketWnd会维护一个报告登记表,当它收到底层WinSock
发出的空闲消息时,就会检索报告登记表,然后直接调用报告者的OnSend函数。所以前文所说的CAsyncSocket
会自动调用OnXxx,实际上是不对的,真正的调用者是CSocketWnd——它是一个CWnd对象,运行在独立的线程中。
以上是CAsyncSocket的原理,而具体代码如下(接收端):
首先在头文件中声明继承自CAsyncSocket的类,并重写三个方法:
#include <afxsock.h> class CMyAsyncSocket : public CAsyncSocket { public: CMyAsyncSocket(void); ~CMyAsyncSocket(void); virtual void OnClose(int nErrorCode); virtual void OnConnect(int nErrorCode); virtual void OnReceive(int nErrorCode); };
在cpp文件中分别实现这三个方法:
void CMyAsyncSocket::OnClose(int nErrorCode) { if ( 0 == nErrorCode) { Close(); } CAsyncSocket::OnClose(nErrorCode); } void CMyAsyncSocket::OnConnect(int nErrorCode) { if (0 == nErrorCode) { //TODO:建立连接后的动作 } CAsyncSocket::OnConnect(nErrorCode); } void CMyAsyncSocket::OnReceive(int nErrorCode) { if (0 == nErrorCode) { //TODO:接受后的处理
char szRcv[513] = "";
int nRcved = Receive(szRcv,15); } CAsyncSocket::OnReceive(nErrorCode); }
接下来,在需要通信的文件中就可以调用这个类:
CMyAsyncSocket m_pClientSocket; AfxSocketInit(); m_pClientSocket.Create(); m_pClientSocket.Connect(_T("IP"),PORT);
在使用过程中有三个地方需要注意:
1.使用前进行AfxSocketInit()初始化;
2.如果出现消息只能接受一次的现象,注意在Receive()后调用父类的OnReceive();
3.在一个方法中,Connect()函数后面不能出现其他代码(这个尚不清楚原因)。