select模型
Winsock分別提供了“套接字模式”和“套接字I/O模型”,可對一個套接字上的I/O行為加以控制。注意:“套接字模式”和“套接字I/O模型”是無關的,套接字模型的出現是為了解決套接字模式存在的某些限制。
Winsock提供兩種套接字模式:鎖定和非鎖定。
套接字I/O模型:Select,WSAAsyncSelect,WSAEventSelect,Overlapped I/O,Completion Port(完成端口)等
前一段時間已經對Winsock的一些基本函數進行了學習,估計和我一樣的初學者已經都看過了這些函數,只是在應用上還存在一些問題,今天我只是在這里介紹一下Select模型,我所寫內容也是借鑒別人的經驗和別人的一些知識,有錯誤的地方還需各位指正。
Select模型:
這個模型的目的就是為了,防止應用程序在套接字處于鎖定模式中時,在一次I/O綁定調用過程中,被迫進入鎖定狀態,同時又是防止在套接字處于非鎖定狀態時,產生WSAEWOULDBLOCK錯誤。
select函數:
int select(
int nfds,fb_set FAR * readfds, fb_set FAR * writefds, fb_set FAR * exceptfds, const struct timeval FAR * timeout);
其中,第一個參數n f d s會被忽略。之所以仍然要提供這個參數,只是為了保持與早期的B e r k e l e y套接字應用程序的兼容。大家可注意到三個f d _ s e t參數:一個用于檢查可讀性(readfds),一個用于檢查可寫性(writefds),另一個用于例外數據(exceptfds)。從根本上說,fb_set數據類型代表著一系列特定套接字的集合。其中, readfds集合包括符合下述任何一個條件的套接字:
■ 有數據可以讀入。
■ 連接已經關閉、重設或中止。
■ 假如已調用了listen,而且一個連接正在建立,那么accept函數調用會成功。
writefds集合包括符合下述任何一個條件的套接字:
■ 有數據可以發出。
■ 如果已完成了對一個非鎖定連接調用的處理,連接就會成功。
最后,exceptfds集合包括符合下述任何一個條件的套接字:
■ 假如已完成了對一個非鎖定連接調用的處理,連接嘗試就會失敗。
■ 有帶外(out-of-band,OOB)數據可供讀取。
例如,假定我們想測試一個套接字是否“可讀”,必須將自己的套接字增添到readfds集合,再等待select函數完成。s e l e c t完成之后,必須判斷自己的套接字是否仍為readfds集合的一部分。若答案是肯定的,便表明該套接字“可讀”,可立即著手從它上面讀取數據。在三個參數中(readfds、writefds和exceptfds),任何兩個都可以是空值( N U L L);但是,至少有一個不能為空值!在任何不為空的集合中,必須包含至少一個套接字句柄;否則, select函數便沒有任何東西可以等待。最后一個參數timeout對應的是一個指針,它指向一個timeval結構,用于決定select最多等待I / O操作完成多久的時間。如timeout是一個空指針,那么select調用會無限期地“鎖定”或停頓下去,直到至少有一個描述符符合指定的條件后結束。對timeval結構的定義如下:
Struct timeval
{
Long tv_sec;
Long tv_usec;
}
其中,tv_sec字段以秒為單位指定等待時間; tv_usec字段則以毫秒為單位指定等待時間。若將超時值設置為( 0 , 0),表明select會立即返回,允許應用程序對select操作進行“輪詢”。出于對性能方面的考慮,應避免這樣的設置。select成功完成后,會在fd_set結構中,返回剛好有未完成的I / O操作的所有套接字句柄的總量。若超過timeval設定的時間,便會返回0。不管由于什么原因,假如select調用失敗,都會返回socket_error。用select對套接字進行監視之前,在自己的應用程序中,必須將套接字句柄分配給一個集合,設置好一個或全部讀、寫以及例外FD_SET結構.
主要涉及到幾個宏操作:
■ FD_CLR(s, *set):從s e t中刪除套接字s。
■ FD_ISSET(s, *set):檢查s是否s e t集合的一名成員;如答案是肯定的是,則返回T R U E。
■ FD_SET(s, *set):將套接字s加入集合s e t。
■ FD_ZERO ( * set ):將s e t初始化成空集合。