簡述: 通知套接口有請求事件發(fā)生. #include <winsock.h>
int PASCAL FAR WSAAsyncSelect ( SOCKET s, HWND hWnd, unsigned int wMsg, long lEvent ); s 標(biāo)識一個需要事件通知的套接口的描述符. hWnd 標(biāo)識一個在網(wǎng)絡(luò)事件發(fā)生時需要接收消息的窗口句柄. wMsg 在網(wǎng)絡(luò)事件發(fā)生時要接收的消息. lEvent 位屏蔽碼,用于指明應(yīng)用程序感興趣的網(wǎng)絡(luò)事件集合. 注釋: 本函數(shù)用來請求Windows Sockets DLL為窗口句柄發(fā)一條消息-無論它何時檢測到由lEvent參數(shù)指明的網(wǎng)絡(luò)事件.要發(fā)送的消息由wMsg參數(shù)標(biāo)明.被通知的套接口由s標(biāo)識. 本函數(shù)自動將套接口設(shè)置為非阻塞模式. lEvent參數(shù)由下表中列出的值組成. 值 意義 FD_READ 欲接收讀準(zhǔn)備好的通知. FD_WRITE 欲接收寫準(zhǔn)備好的通知. FD_OOB 欲接收帶邊數(shù)據(jù)到達(dá)的通知. FD_ACCEPT 欲接收將要連接的通知. FD_CONNECT 欲接收已連接好的通知. FD_CLOSE 欲接收套接口關(guān)閉的通知. 啟動一個WSAAsyncSelect()將使為同一個套接口啟動的所有先前的WSAAsyncSelect()作廢. 例如,要接收讀寫通知,應(yīng)用程序必須同時用FD_READ和FD_WRITE調(diào)用WSAAsyncSelect(),如下: rc = WSAAsyncSelect(s, hWnd, wMsg, FD_READ|FD_WRITE); 對不同的事件區(qū)分不同的消息是不可能的.下面的代碼將不會工作;第二個調(diào)用將會使第一次調(diào)用的作用失效,只有FD_WRITE會通過wMsg2消息通知到. rc = WSAAsyncSelect(s, hWnd, wMsg1, FD_READ); rc = WSAAsyncSelect(s, hWnd, wMsg2, FD_WRITE);
如果要取消所有的通知,也就是指出Windows Sockets的實現(xiàn)不再在套接口上發(fā)送任何和網(wǎng)絡(luò)事件相關(guān)的消息,則lEvent應(yīng)置為0. rc = WSAAsyncSelect(s, hWnd, 0, 0); 盡管在本例中,WSAAsyncSelect()立即使傳給該套接口的事件消息無效, 仍有可能有消息等在應(yīng)用程序的消息隊列中.應(yīng)用程序因此也必須仍準(zhǔn)備好接收網(wǎng)絡(luò)消息-即使消息作廢.用closesocket()關(guān)閉一個套接口也同樣使WSAAsyncSelect()發(fā)送的消息作廢,但在closesocke()之前隊列中的消息仍然起作用. 由于一個已調(diào)用accept()的套接口和用來接收它的偵聽套接口有同樣的屬性, 任何為偵聽套接口設(shè)置的的WSAAsyncSelect()事件也同樣對已接收的套接口起作用.例如, 如果一個偵聽的套接口有WSAAsyncSelect()事件FD_ACCEPT,FD_READ,FD_WRITE, 則任何在那個偵聽的套接口上接收的套接口將也有FD_ACCEPT,FD_READ,FD_WRITE事件,以及同樣的wMsg的值.若需要不同的wMsg及事件,應(yīng)用程序應(yīng)調(diào)用WSAAsyncSelect(),將已接收的套接口和想要發(fā)送的新消息作為參數(shù)傳遞. 當(dāng)某一套接口s上發(fā)生了一個已命名的網(wǎng)絡(luò)事件,應(yīng)用程序窗口hWnd會接收到消息wMsg.wParam參數(shù)標(biāo)識了網(wǎng)絡(luò)事件發(fā)生的套接口.lParam的低字指明了發(fā)生的網(wǎng)絡(luò)事件.lParam的高字則含有一個錯誤代碼.該錯誤代碼可以是winsock.h中定義的任何錯誤. 錯誤代碼和事件可以通過WSAGETSELECTERRORH和WSAGETSELECTEVENT宏從lParam中取出.定義如下: #define WSAGETSELECTERROR(lParam) HIWORD(lParam) #define WSAGETSELECTEVENT(lParam) LOWORD(lParam) 注意:在accept()調(diào)用和為改變事件或wMsg的WSAAsyncSelect()調(diào)用中有一個計時窗口.應(yīng)用程序如果需要給偵聽的和調(diào)用過accept()的套接口以不同的wMsg,它就應(yīng)該在偵聽的套接口上請求FD_ACCEPT事件,然后在accept()調(diào)用后設(shè)置相應(yīng)的事件.由于FD_ACCEPT從不發(fā)送給已連接的套接口,而FD_READ,FD_WRITE,FD_OOB及FD_CLOSE也從不發(fā)送給偵聽套接口,所以不會產(chǎn)生困難. 使用以上的宏將最大限度的提高應(yīng)用程序的可移植性. 返回的可能網(wǎng)絡(luò)事件如下: 值 意義 FD_READ 套接口s準(zhǔn)備讀 FD_WRITE 套接口s準(zhǔn)備寫 FD_OOB 帶外數(shù)據(jù)準(zhǔn)備好在套接口s上讀. FD_ACCEPT 套接口s準(zhǔn)備接收新的將要到來的連接. FD_CONNECT 套接口s上的連接完成. FD_CLOSE 由套接口s標(biāo)識的連接已關(guān)閉.
返回值: 0 若應(yīng)用程序感興趣的網(wǎng)絡(luò)事件的聲明成功. SOCKET_ERROR 否則.可通過調(diào)用WSAGetLastError()返回特定的錯誤代碼.
評價: 盡管WSAAsyncSelect()可以以多個事件的組合來調(diào)用,應(yīng)用程序窗口還是會為每個網(wǎng)絡(luò)事件接收一條消息. 如同select()函數(shù),WSAAsyncSelect()會被頻繁地調(diào)用來決定,何時一次數(shù)據(jù)轉(zhuǎn)移操作(send()或recv())可以啟動,并且可以立刻成功.盡管如此,健壯的應(yīng)用程序必須做好這樣的準(zhǔn)備, 即它可能接收到消息及啟動了一個會立即返回WSAEWOULDBLOCK的Windows Sockets API調(diào)用.例如,下列的事件序列是可能的: (i) 數(shù)據(jù)到達(dá)套接口s;Windows Sockets傳遞WSAAsyncSelect消息. (ii) 應(yīng)用程序處理其它一些消息. (iii) 在處理過程中,應(yīng)用程序啟動了ioctlsocket(s,FIONREAD...)并且注意到有數(shù)據(jù)準(zhǔn)備好讀. (iv) 應(yīng)用程序啟動recv(s,...)來讀數(shù)據(jù). (v) 應(yīng)用程序循環(huán)處理下一條消息,最終到達(dá)WSAAsyncSelect消息,表示數(shù)據(jù)已準(zhǔn)備好讀. (vi) 應(yīng)用程序啟動recv(s,...),但失敗并有錯誤WSAEWOULDBLOCK. 其它的事件序列也是可能的. Windows Sockets DLL不會不斷地為某一特定的網(wǎng)絡(luò)事件向一個應(yīng)用程序發(fā)送消息. 如果已成功地向應(yīng)用程序窗口發(fā)送了一特定事件的通知,對該應(yīng)用程序窗口將不再為該網(wǎng)絡(luò)事件發(fā)消息,直到應(yīng)用程序調(diào)用函數(shù)隱含地重新通知該網(wǎng)絡(luò)事件. 事件 重新通知函數(shù) FD_READ recv()或recvfrom() FD_WRITE send()或sendto() FD_OOB recv() FD_ACCEPT accept() FD_CONNECT 無 FD_CLOSE 無 任何對重新通知函數(shù)的調(diào)用,即使失敗,也會達(dá)到為相關(guān)事件發(fā)重新通知消息的效果. 對FD_READ,FD_OOB和FD_ACCEPT事件,消息傳遞是"水平觸發(fā)"(level-triggered)的.這意味著,若調(diào)用了重新通知函數(shù)并且相關(guān)的事件對該調(diào)用仍有效,WSAAsyncSelect()消息就將傳給應(yīng)用程序.這為應(yīng)用程序提供了事件驅(qū)動以及不必考慮在任一時刻到達(dá)的數(shù)據(jù)量的能力.考慮下列序列: (i) Windows Sockets DLL在套接口s上接收100字節(jié)的數(shù)據(jù)并傳遞一個FD_READ消息. (ii) 應(yīng)用程序啟動recv(s,buffptr,50,0)接收50字節(jié). (iii) 由于仍有數(shù)據(jù)未讀,Windows Sockets DLL發(fā)送另一個FD_READ消息. 根據(jù)以上語義,應(yīng)用程序不必在收到FD_READ消息時讀進(jìn)所有可讀的數(shù)據(jù)-對應(yīng)于每一FD_READ消息進(jìn)行一次recv()調(diào)用是恰當(dāng)?shù)?如果應(yīng)用程序為一個FD_READ消息而啟動了多個recv()調(diào)用,它將接收到多個FD_READ消息.這樣的應(yīng)用程序可能希望在開始recv()調(diào)用( 通過不為FD_READ事件置位的WSAAsyncSelect()函數(shù)調(diào)用)之前關(guān)閉FD_READ消息. 如果在應(yīng)用程序初次調(diào)用WSAAsyncSelect()或當(dāng)調(diào)用了重新通知函數(shù)時,有一個事件為真, 則會發(fā)送一個相應(yīng)的消息.例如,若應(yīng)用程序調(diào)用listen(),就會試圖進(jìn)行連接,然后應(yīng)用程序調(diào)用WSAAsyncSelect()聲明它需要為套接口接收FD_ACCEPT消息,Windows Sockets的實現(xiàn)就會立即傳遞一個FD_ACCEPT消息. FD_WRITE事件處理起來稍有不同.FD_WRITE消息是在套接口第一次用connect()連接或由accept()接受,并且在send()或sendto()以WSAWOULDBLOCK錯誤失敗后緩沖區(qū)空閑時發(fā)送的.因此,應(yīng)用程序可以假設(shè)發(fā)送可能在第一次FD_WRITE消息時開始,并持續(xù)到一次返回WSAEWOULDBLOCK的發(fā)送. 在這樣的失敗后,應(yīng)用程序?qū)⒈煌ㄖ?FD_WRITE消息的發(fā)送又將可能. FD_OOB事件只用在當(dāng)套接口配置成獨立接收帶外數(shù)據(jù)時.如果一個套接口被配置成接收感興趣的帶外數(shù)據(jù)狀態(tài),帶外數(shù)據(jù)將和普通數(shù)據(jù)等同視之,并且應(yīng)用程序應(yīng)該注冊它感興趣的方面,然后將接收FD_READ事件,而不是FD_OOB事件.應(yīng)用程序可以設(shè)置或監(jiān)控帶外數(shù)據(jù)處理的方法(通過使用setsockopt()或getsockopt()函數(shù),及SO_OOBINLINE選項). 在FD_CLOSE消息中的錯誤代碼指出套接口的關(guān)閉是正常的還是異常的.如果錯誤代碼是0,則關(guān)閉是正常的;若錯誤代碼是WSAECONNRESET,則套接口的虛套接口將被重置.這些只對SOCK_STREAM類型的套接口起作用. FD_CLOSE消息在相應(yīng)套接口的虛電路關(guān)閉指令接收到時發(fā)送.在TCP術(shù)語中,這意味著FD_CLOSE在連接進(jìn)入了FIN WAIT或CLOSE WAIT狀態(tài)時發(fā)送.這是遠(yuǎn)端對發(fā)送方進(jìn)行了shutdown()調(diào)用或closesocket()調(diào)用的結(jié)果. 請注意你的應(yīng)用程序?qū)⒅粫盏紽D_CLOSE消息來指出虛電路的關(guān)閉.它不會收到FD_READ消息來表示該狀況.
錯誤代碼: WSANOTINITIALISED 在使用本API前必須進(jìn)行一次成功的WSAStartup()調(diào)用. WSAENETDOWN WINDOWS SOCKETS實現(xiàn)已檢測到網(wǎng)絡(luò)子系統(tǒng)故障. WSAEINVAL 指出指定的參數(shù)之一是非法的. WSAEINPROGRESS 一個阻塞的Windows Sockets操作正在進(jìn)行. 附加的錯誤代碼可能在應(yīng)用程序窗口接收到消息時被置.這些代碼可以用WSAGETSELECTERROR宏從lParam中取出.對應(yīng)于每個網(wǎng)絡(luò)事件的可能錯誤代碼為: 事件:FD_CONNECT WSAEADDRINUSE 給定的地址已被使用. WSAEADDRNOTAVAIL 指定的地址在本地機(jī)器不能使用. WSAEAFNOSUPPORT 指定族的地址不能和本套接口同時使用. WSAECONNREFUSED 連接的嘗試被拒絕. WSAEDESTADDRREQ 需要一個目的地址. WSAEFAULT namelen參數(shù)不正確. WSAEINVAL 套接口已經(jīng)約束到一個地址. WSAEISCONN 套接口已經(jīng)連接. WSAEMFILE 沒有可用的文件描述符. WSAENETUNREACH 此時網(wǎng)絡(luò)不能從該主機(jī)訪問. WSAENOBUFS 無可用的緩沖區(qū)空間.套接口不能連接. WSAENOTCONN 套接口沒有連接. WSAENOTSOCK 該描述符是文件,不是套接口. WSAETIMEDOUT 試圖連接超時,未建立連接. 事件:FD_CLOSE WSAENETDOWN WINDOWS SOCKETS實現(xiàn)已檢測到網(wǎng)絡(luò)子系統(tǒng)故障. WSAECONNRESET 連接由遠(yuǎn)端重建. WSAECONNABORTED 由于超時或其它失敗放棄連接. 事件:FD_READ 事件:FD_WRITE 事件:FD_OOB 事件:FD_ACCEPT WSAENETDOWN WINDOWS SOCKETS實現(xiàn)已檢測到網(wǎng)絡(luò)子系統(tǒng)故障.
關(guān)于Windows Sockets提供者的說明: Windows Sockets的提供者應(yīng)確保消息可以成功地傳給應(yīng)用程序.如果PostMessag()操作失敗,Windows Sockets的實現(xiàn)必須重發(fā)該消息-只要窗口存在. Windows Sockets提供者應(yīng)使用WSAMAKESELECTREPLY宏來構(gòu)造消息中的lParam參數(shù). 當(dāng)套接口關(guān)閉時,Windows Sockets提供者應(yīng)清除所有保留下來要發(fā)送給應(yīng)用程序窗口的消息.然而應(yīng)用程序必須準(zhǔn)備好接收,放棄任何在closesocket()之前可能已經(jīng)發(fā)送的消息.
|