WSAEventSelect是提供以網(wǎng)絡(luò)事件的異步方式通知模型,為套接字注冊(cè)網(wǎng)絡(luò)事件后等待通知,故是被動(dòng)發(fā)生的。具體流程可以參照如下圖

MSDN的網(wǎng)絡(luò)事件
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 |
能夠熟練運(yùn)用WSAEventSelect必須注意以下5點(diǎn)
1、 必須創(chuàng)建事件表和套接字表,并把監(jiān)聽(tīng)套接字放到表的首位。
① 、對(duì)于TCP而言:關(guān)聯(lián)監(jiān)聽(tīng)套接字必須處理 (FD_ACCEPT或者FD_CONNECT) 和FD_CLOSE事件。監(jiān)聽(tīng)套接字不參與通信 ,故FD_WRITE 不需要關(guān)聯(lián)。而由通信套接字創(chuàng)建的網(wǎng)絡(luò)事件則按連接順序放到事件表和套接字表。這時(shí)需要處理的事件有FD_READ 和FD_CLOSE 和FD_WRITE。
② 對(duì)于UDP不存在連接,故只需要處理FD_WRITE 或者FD_READ 就可以、
2、 什么是“WSAEnumNetworkEvents函數(shù)會(huì)自動(dòng)重置受信事件”!
大家知道WSAEnumNetworkEvents是查詢套接字上的發(fā)生的網(wǎng)絡(luò)事件和出錯(cuò)代碼。然而對(duì)于第二個(gè)參數(shù),hEventObject。其實(shí)重置受信事件則是清除系統(tǒng)內(nèi)部的網(wǎng)絡(luò)事件記錄。當(dāng)調(diào)用一次WSAEnumNetworkEvents 返回event(WSAEVENT)事件 ,接下來(lái)就應(yīng)該是處理該事件的代碼。如果沒(méi)有清除該事件對(duì)象的網(wǎng)絡(luò)事件,那么有可能出現(xiàn)不斷處理的該事件的結(jié)果。
3、 一個(gè)連接關(guān)閉時(shí)應(yīng)該在事件表和套接字表中刪除該套接字和事件。并在事件表和套接字表上把其他的前移。以便省留更多的位置給新的連接,要知道在不創(chuàng)建線程池的情況下,WSAEventSelect模型最多也就是能支持64 -1個(gè)連接、
4、 如果同時(shí)有幾個(gè)事件對(duì)象受信WSAWaitForMultipleEvents函數(shù)只返回,最前面的一個(gè),因此要從最前面受信事件開(kāi)始逐個(gè)檢查后面的受信事件,此時(shí)使用WSAWaitForMultipleEvents就要設(shè)置超時(shí)返回了,也就是dwTimeout參數(shù)、
5、 IP地址與套接字的關(guān)聯(lián)問(wèn)題,在接受眾多連接后想要向某一連接發(fā)送數(shù)據(jù),則需要找到該IP地址(或者主機(jī)名也行)與服務(wù)器建立的通信套接字。那么套接字表就不能是簡(jiǎn)單的套接字集合了,或許可以這樣
struct SOCKET_ADDR //用來(lái)記錄客戶區(qū)信息的對(duì)象
{
SOCKET s;
sockaddr_in in_addr;
};
而套接字表為
SOCKET_ADDR sockArray[64];
::更加受歡迎的做法就是符合面向?qū)ο筇捉幼宙湵砹恕?/span>
typedef struct _SOCKET_OBJ
{
SOCKET s; // 套節(jié)字句柄
HANDLE event; // 與此套節(jié)字相關(guān)聯(lián)的事件對(duì)象句柄
sockaddr_in addrRemote; // 客戶端地址信息
_SOCKET_OBJ *pNext;// 指向下一個(gè)SOCKET_OBJ對(duì)象,為的是連成一個(gè)表
} SOCKET_OBJ, *PSOCKET_OBJ;
posted on 2011-08-24 21:25
Yu_ 閱讀(1117)
評(píng)論(0) 編輯 收藏 引用 所屬分類:
網(wǎng)絡(luò)編程