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