WSAAsyncSelect()函數詳解

簡述:
        通知套接口有請求事件發生.
        #include <winsock.h>

 int PASCAL FAR WSAAsyncSelect ( SOCKET s, HWND hWnd,
                        unsigned int wMsg,  long lEvent );
       
        s   標識一個需要事件通知的套接口的描述符.
        hWnd    標識一個在網絡事件發生時需要接收消息的窗口句柄.
        wMsg    在網絡事件發生時要接收的消息.
        lEvent  位屏蔽碼,用於指明應用程序感興趣的網絡事件集合.
       
註釋:
        本函數用來請求Windows Sockets DLL爲窗口句柄發一條消息-無論它何時檢測到由lEvent參數指明的網絡事件.要發送的消息由wMsg參數標明.被通知的套接口由s標識.
        本函數自動將套接口設置爲非阻塞模式.
        lEvent參數由下表中列出的值組成.
        值      意義
        FD_READ 欲接收讀準備好的通知.
        FD_WRITE    欲接收寫準備好的通知.
        FD_OOB  欲接收帶邊數據到達的通知.
        FD_ACCEPT   欲接收將要連接的通知.
        FD_CONNECT  欲接收已連接好的通知.
        FD_CLOSE    欲接收套接口關閉的通知.
       
        啓動一個WSAAsyncSelect()將使爲同一個套接口啓動的所有先前的WSAAsyncSelect()作廢. 例如,要接收讀寫通知,應用程序必須同時用FD_READ和FD_WRITE調用WSAAsyncSelect(),如下:
        rc = WSAAsyncSelect(s, hWnd, wMsg, FD_READ|FD_WRITE);
        對不同的事件區分不同的消息是不可能的.下面的代碼將不會工作;第二個調用將會使第一次調用的作用失效,只有FD_WRITE會通過wMsg2消息通知到.
                        rc = WSAAsyncSelect(s, hWnd, wMsg1, FD_READ);
                        rc = WSAAsyncSelect(s, hWnd, wMsg2, FD_WRITE);

        如果要取消所有的通知,也就是指出Windows Sockets的實現不再在套接口上發送任何和網絡事件相關的消息,則lEvent應置爲0.
                        rc = WSAAsyncSelect(s, hWnd, 0, 0);
       
        儘管在本例中,WSAAsyncSelect()立即使傳給該套接口的事件消息無效, 仍有可能有消息等在應用程序的消息隊列中.應用程序因此也必須仍準備好接收網絡消息-即使消息作廢.用closesocket()關閉一個套接口也同樣使WSAAsyncSelect()發送的消息作廢,但在closesocke()之前隊列中的消息仍然起作用.
        由於一個已調用accept()的套接口和用來接收它的偵聽套接口有同樣的屬性, 任何爲偵聽套接口設置的的WSAAsyncSelect()事件也同樣對已接收的套接口起作用.例如, 如果一個偵聽的套接口有WSAAsyncSelect()事件FD_ACCEPT,FD_READ,FD_WRITE, 則任何在那個偵聽的套接口上接收的套接口將也有FD_ACCEPT,FD_READ,FD_WRITE事件,以及同樣的wMsg的值.若需要不同的wMsg及事件,應用程序應調用WSAAsyncSelect(),將已接收的套接口和想要發送的新消息作爲參數傳遞.
        當某一套接口s上發生了一個已命名的網絡事件,應用程序窗口hWnd會接收到消息wMsg.wParam參數標識了網絡事件發生的套接口.lParam的低字指明瞭發生的網絡事件.lParam的高字則含有一個錯誤代碼.該錯誤代碼可以是winsock.h中定義的任何錯誤.
        錯誤代碼和事件可以通過WSAGETSELECTERRORH和WSAGETSELECTEVENT宏從lParam中取出.定義如下:
                  #define WSAGETSELECTERROR(lParam)            HIWORD(lParam)
                  #define WSAGETSELECTEVENT(lParam)            LOWORD(lParam)
注意:在accept()調用和爲改變事件或wMsg的WSAAsyncSelect()調用中有一個計時窗口.應用程序如果需要給偵聽的和調用過accept()的套接口以不同的wMsg,它就應該在偵聽的套接口上請求FD_ACCEPT事件,然後在accept()調用後設置相應的事件.由於FD_ACCEPT從不發送給已連接的套接口,而FD_READ,FD_WRITE,FD_OOB及FD_CLOSE也從不發送給偵聽套接口,所以不會產生困難.
        使用以上的宏將最大限度的提高應用程序的可移植性.
        返回的可能網絡事件如下:
        值      意義
        FD_READ 套接口s準備讀
        FD_WRITE    套接口s準備寫
        FD_OOB  帶外數據準備好在套接口s上讀.
        FD_ACCEPT   套接口s準備接收新的將要到來的連接.
        FD_CONNECT  套接口s上的連接完成.
        FD_CLOSE    由套接口s標識的連接已關閉.

返回值:
        0           若應用程序感興趣的網絡事件的聲明成功.
        SOCKET_ERROR    否則.可通過調用WSAGetLastError()返回特定的錯誤代碼.

評價:
        儘管WSAAsyncSelect()可以以多個事件的組合來調用,應用程序窗口還是會爲每個網絡事件接收一條消息.
        如同select()函數,WSAAsyncSelect()會被頻繁地調用來決定,何時一次數據轉移操作(send()或recv())可以啓動,並且可以立刻成功.儘管如此,健壯的應用程序必須做好這樣的準備, 即它可能接收到消息及啓動了一個會立即返回WSAEWOULDBLOCK的Windows Sockets API調用.例如,下列的事件序列是可能的:
        (i) 數據到達套接口s;Windows Sockets傳遞WSAAsyncSelect消息.
        (ii)    應用程序處理其它一些消息.
        (iii)   在處理過程中,應用程序啓動了ioctlsocket(s,FIONREAD...)並且注意到有數據準備好讀.
        (iv)    應用程序啓動recv(s,...)來讀數據.
        (v) 應用程序循環處理下一條消息,最終到達WSAAsyncSelect消息,表示數據已準備好讀.
        (vi)    應用程序啓動recv(s,...),但失敗並有錯誤WSAEWOULDBLOCK.
        其它的事件序列也是可能的.
        windows Sockets DLL不會不斷地爲某一特定的網絡事件向一個應用程序發送消息. 如果已成功地嚮應用程序窗口發送了一特定事件的通知,對該應用程序窗口將不再爲該網絡事件發消息,直到應用程序調用函數隱含地重新通知該網絡事件.
        事件        重新通知函數
        FD_READ recv()或recvfrom()
        FD_WRITE    send()或sendto()
        FD_OOB  recv()
        FD_ACCEPT   accept()
        FD_CONNECT  無
        FD_CLOSE    無
        任何對重新通知函數的調用,即使失敗,也會達到爲相關事件發重新通知消息的效果.
        對FD_READ,FD_OOB和FD_ACCEPT事件,消息傳遞是"水平觸發"(level-triggered)的.這意味着,若調用了重新通知函數並且相關的事件對該調用仍有效,WSAAsyncSelect()消息就將傳給應用程序.這爲應用程序提供了事件驅動以及不必考慮在任一時刻到達的數據量的能力.考慮下列序列:
        (i) windows Sockets DLL在套接口s上接收100字節的數據並傳遞一個FD_READ消息.
        (ii)    應用程序啓動recv(s,buffptr,50,0)接收50字節.
        (iii)   由於仍有數據未讀,Windows Sockets DLL發送另一個FD_READ消息.
       
        根據以上語義,應用程序不必在收到FD_READ消息時讀進所有可讀的數據-對應於每一FD_READ消息進行一次recv()調用是恰當的.如果應用程序爲一個FD_READ消息而啓動了多個recv()調用,它將接收到多個FD_READ消息.這樣的應用程序可能希望在開始recv()調用( 通過不爲FD_READ事件置位的WSAAsyncSelect()函數調用)之前關閉FD_READ消息.
        如果在應用程序初次調用WSAAsyncSelect()或當調用了重新通知函數時,有一個事件爲真, 則會發送一個相應的消息.例如,若應用程序調用listen(),就會試圖進行連接,然後應用程序調用WSAAsyncSelect()聲明它需要爲套接口接收FD_ACCEPT消息,Windows Sockets的實現就會立即傳遞一個FD_ACCEPT消息.
        FD_WRITE事件處理起來稍有不同.FD_WRITE消息是在套接口第一次用connect()連接或由accept()接受,並且在send()或sendto()以WSAWOULDBLOCK錯誤失敗後緩衝區空閒時發送的.因此,應用程序可以假設發送可能在第一次FD_WRITE消息時開始,並持續到一次返回WSAEWOULDBLOCK的發送. 在這樣的失敗後,應用程序將被通知,FD_WRITE消息的發送又將可能.
        FD_OOB事件只用在當套接口配置成獨立接收帶外數據時.如果一個套接口被配置成接收感興趣的帶外數據狀態,帶外數據將和普通數據等同視之,並且應用程序應該註冊它感興趣的方面,然後將接收FD_READ事件,而不是FD_OOB事件.應用程序可以設置或監控帶外數據處理的方法(通過使用setsockopt()或getsockopt()函數,及SO_OOBINLINE選項).
        在FD_CLOSE消息中的錯誤代碼指出套接口的關閉是正常的還是異常的.如果錯誤代碼是0,則關閉是正常的;若錯誤代碼是WSAECONNRESET,則套接口的虛套接口將被重置.這些只對SOCK_STREAM類型的套接口起作用.
        FD_CLOSE消息在相應套接口的虛電路關閉指令接收到時發送.在TCP術語中,這意味着FD_CLOSE在連接進入了FIN WAIT或CLOSE WAIT狀態時發送.這是遠端對發送方進行了shutdown()調用或closesocket()調用的結果.
        請注意你的應用程序將只會收到FD_CLOSE消息來指出虛電路的關閉.它不會收到FD_READ消息來表示該狀況.

錯誤代碼:
        WSANOTINITIALISED       在使用本API前必須進行一次成功的WSAStartup()調用.
        WSAENETDOWN     WINDOWS SOCKETS實現已檢測到網絡子系統故障. 
        WSAEINVAL           指出指定的參數之一是非法的.
        WSAEINPROGRESS      一個阻塞的Windows Sockets操作正在進行.
        附加的錯誤代碼可能在應用程序窗口接收到消息時被置.這些代碼可以用WSAGETSELECTERROR宏從lParam中取出.對應於每個網絡事件的可能錯誤代碼爲:
        事件:FD_CONNECT
        WSAEADDRINUSE       給定的地址已被使用.
        WSAEADDRNOTAVAIL        指定的地址在本地機器不能使用.
        WSAEAFNOSUPPORT     指定族的地址不能和本套接口同時使用.
        WSAECONNREFUSED     連接的嘗試被拒絕.
        WSAEDESTADDRREQ     需要一個目的地址.
        WSAEFAULT           namelen參數不正確.
        WSAEINVAL           套接口已經約束到一個地址.
        WSAEISCONN          套接口已經連接.
        WSAEMFILE           沒有可用的文件描述符.
        WSAENETUNREACH      此時網絡不能從該主機訪問.
        WSAENOBUFS          無可用的緩衝區空間.套接口不能連接.
        WSAENOTCONN     套接口沒有連接.
        WSAENOTSOCK     該描述符是文件,不是套接口.
        WSAETIMEDOUT        試圖連接超時,未建立連接.
        事件:FD_CLOSE
        WSAENETDOWN     WINDOWS SOCKETS實現已檢測到網絡子系統故障. 
        WSAECONNRESET       連接由遠端重建.
        WSAECONNABORTED     由於超時或其它失敗放棄連接.
        事件:FD_READ
        事件:FD_WRITE
        事件:FD_OOB
        事件:FD_ACCEPT
        WSAENETDOWN     WINDOWS SOCKETS實現已檢測到網絡子系統故障. 

關於Windows Sockets提供者的說明:
        windows Sockets的提供者應確保消息可以成功地傳給應用程序.如果PostMessag()操作失敗,Windows Sockets的實現必須重發該消息-只要窗口存在.
        windows Sockets提供者應使用WSAMAKESELECTREPLY宏來構造消息中的lParam參數.
        當套接口關閉時,Windows Sockets提供者應清除所有保留下來要發送給應用程序窗口的消息.然而應用程序必須準備好接收,放棄任何在closesocket()之前可能已經發送的消息.

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章