CSDN博客

img ccnuxjg

深入了解异步套接字

发表于2001/11/21 20:10:00  752人阅读

分类: Visual C++ Network

 

深入了解异步套接字

 CAsyncSocket异步套接字是MFC封装的SOCKET的类。

只有在三种条件下,才会发出FD_WRITE通知:
1、使用connect或WSAConnect,一个套接字首次建立了连接;
2、使用accept或WSAAccept,套接字被接受以后;
3、若send、WSASend、sendto或WSASendTo操作失败,返回了WSAEWOULDBLOCK错误

,而且缓冲区的空间变得可用。
因此,一个应用程序自收到FD_WRITE消息开始,便认为自己必然能在一个套接字

上发出数据。
所以系统是通过掩码(例:FD_WRITE)来通知异步套接字消息响应函数的。


在实例程序中,经过设置断点调试得知系统会在如下的函数中处理异步消息:
void PASCAL CAsyncSocket::DoCallBack(WPARAM wParam, LPARAM lParam)
{
 if (wParam == 0 && lParam == 0)
  return;

 // Has the socket be closed?
 CAsyncSocket* pSocket =

CAsyncSocket::LookupHandle((SOCKET)wParam, TRUE);

 // If yes ignore message
 if (pSocket != NULL)
  return;

 pSocket = CAsyncSocket::LookupHandle((SOCKET)wParam, FALSE);
 if (pSocket == NULL)
 {
  // Must be in the middle of an Accept call
  pSocket = CAsyncSocket::LookupHandle(INVALID_SOCKET,

FALSE);
  ASSERT(pSocket != NULL);
  pSocket->m_hSocket = (SOCKET)wParam;
  CAsyncSocket::DetachHandle(INVALID_SOCKET, FALSE);
  CAsyncSocket::AttachHandle(pSocket->m_hSocket, pSocket,

FALSE);
 }

 int nErrorCode = WSAGETSELECTERROR(lParam);
 switch (WSAGETSELECTEVENT(lParam))
 {
 case FD_READ://如果来的异步消息FD_READ,就准备触发OnReceive() 函数
  {
   DWORD nBytes;
   if (!pSocket->IOCtl(FIONREAD, &nBytes))
    nErrorCode = WSAGetLastError();
   if (nBytes != 0 || nErrorCode != 0)
    pSocket->OnReceive(nErrorCode);
  }
  break;
 case FD_WRITE:
  pSocket->OnSend(nErrorCode);
  break;
 case FD_OOB:
  pSocket->OnOutOfBandData(nErrorCode);
  break;
 case FD_ACCEPT:
  pSocket->OnAccept(nErrorCode);
  break;
 case FD_CONNECT:
  pSocket->OnConnect(nErrorCode);
  break;
 case FD_CLOSE:
  pSocket->OnClose(nErrorCode);
  break;
 }
}
从以上的代码可以看出,MFC是通过DoCallBack()这个函数来处理异步网络事件。
当我们用
 AsyncSelect(FD_READ);的话,一旦成功,系统将会触发函数

OnReceive(nErrorCode);
 同样FD_WRITE对应函数OnSend(nErrorCode);
 同样FD_ACCEPT对应函数OnAccept(nErrorCode);
 同样FD_CONNECT对应函数OnConnect(nErrorCode);
 同样FD_CLOSE对应函数OnClose(nErrorCode);
 同样FD_OOB对应函数OnOutOfBandData(nErrorCode);

0 0

相关博文

我的热门文章

img
取 消
img