WSAEventSelect是提供以網絡事件的異步方式通知模型,為套接字注冊網絡事件后等待通知,故是被動發生的。具體流程可以參照如下圖

MSDN的網絡事件
Network event |
Re-enabling function |
FD_READ |
The recv, recvfrom, WSARecv, WSARecvEx, or WSARecvFrom function. |
FD_WRITE |
The send, sendto, WSASend, or WSASendTo function. |
FD_OOB |
The recv, recvfrom, WSARecv, WSARecvEx, or WSARecvFrom function. |
FD_ACCEPT |
The accept, AcceptEx, or WSAAccept function unless the error code returned is WSATRY_AGAIN indicating that the condition function returned CF_DEFER. |
FD_CONNECT |
None. |
FD_CLOSE |
None. |
FD_QOS |
The WSAIoctl function with command SIO_GET_QOS. |
FD_GROUP_QOS |
Reserved. |
FD_ROUTING_ INTERFACE_CHANGE |
The WSAIoctl function with command SIO_ROUTING_INTERFACE_CHANGE. |
FD_ADDRESS_ LIST_CHANGE |
The WSAIoctl function with command SIO_ADDRESS_LIST_CHANGE |
能夠熟練運用WSAEventSelect必須注意以下5點
1、 必須創建事件表和套接字表,并把監聽套接字放到表的首位。
① 、對于TCP而言:關聯監聽套接字必須處理 (FD_ACCEPT或者FD_CONNECT) 和FD_CLOSE事件。監聽套接字不參與通信 ,故FD_WRITE 不需要關聯。而由通信套接字創建的網絡事件則按連接順序放到事件表和套接字表。這時需要處理的事件有FD_READ 和FD_CLOSE 和FD_WRITE。
② 對于UDP不存在連接,故只需要處理FD_WRITE 或者FD_READ 就可以、
2、 什么是“WSAEnumNetworkEvents函數會自動重置受信事件”!
大家知道WSAEnumNetworkEvents是查詢套接字上的發生的網絡事件和出錯代碼。然而對于第二個參數,hEventObject。其實重置受信事件則是清除系統內部的網絡事件記錄。當調用一次WSAEnumNetworkEvents 返回event(WSAEVENT)事件 ,接下來就應該是處理該事件的代碼。如果沒有清除該事件對象的網絡事件,那么有可能出現不斷處理的該事件的結果。
3、 一個連接關閉時應該在事件表和套接字表中刪除該套接字和事件。并在事件表和套接字表上把其他的前移。以便省留更多的位置給新的連接,要知道在不創建線程池的情況下,WSAEventSelect模型最多也就是能支持64 -1個連接、
4、 如果同時有幾個事件對象受信WSAWaitForMultipleEvents函數只返回,最前面的一個,因此要從最前面受信事件開始逐個檢查后面的受信事件,此時使用WSAWaitForMultipleEvents就要設置超時返回了,也就是dwTimeout參數、
5、 IP地址與套接字的關聯問題,在接受眾多連接后想要向某一連接發送數據,則需要找到該IP地址(或者主機名也行)與服務器建立的通信套接字。那么套接字表就不能是簡單的套接字集合了,或許可以這樣
struct SOCKET_ADDR //用來記錄客戶區信息的對象
{
SOCKET s;
sockaddr_in in_addr;
};
而套接字表為
SOCKET_ADDR sockArray[64];
::更加受歡迎的做法就是符合面向對象套接字鏈表了。
typedef struct _SOCKET_OBJ
{
SOCKET s; // 套節字句柄
HANDLE event; // 與此套節字相關聯的事件對象句柄
sockaddr_in addrRemote; // 客戶端地址信息
_SOCKET_OBJ *pNext;// 指向下一個SOCKET_OBJ對象,為的是連成一個表
} SOCKET_OBJ, *PSOCKET_OBJ;
posted on 2011-08-24 21:25
Yu_ 閱讀(1117)
評論(0) 編輯 收藏 引用 所屬分類:
網絡編程