select函式的作用:
select()在socket程式設計中還是比較重要的,可是對於初學socket的人來說都不太愛用select()寫程式,他們只是習慣寫諸如 conncet()、accept()、recv()或recvfrom這樣的阻塞程式(所謂阻塞方式block,顧名思義,就是程序或是執行緒執行到這些函式時必須等待某個事件發生,如果事件沒有發生,程序或執行緒就被阻塞,函式不能立即返回)。可是使用select()就可以完成非阻塞(所謂非阻塞方式non-block,就是程序或執行緒執行此函式時不必非要等待事件的發生,一旦執行肯定返回,以返回值的不同來反映函式的執**況。如果事件發生則與阻塞方式相同,若事件沒有發生則返回乙個**來告知事件未發生,而程序或執行緒繼續執行,所以效率高)方式工作的程式,它能夠監視我們需要監視的檔案描述符的變化情況——讀寫或是異常。
select函式格式:
select()函式的格式(所說的是unix系統下的berkeley socket程式設計,和windows下的有區別,一會兒說明):
unix系統下解釋:
int select(int maxfdp, fd_set* readfds, fd_set* writefds, fd_set* errorfds, struct timeval* timeout);
先說明兩個結構體:
第一:struct fd_set可以理解為乙個集合,這個集合中存放的是檔案描述符(file descriptor),即檔案控制代碼,這可以是我們所說的普通意義的檔案,當然unix下任何裝置、管道、fifo等都是檔案形式,全部包括在內,所以,毫無疑問,乙個socket就是乙個檔案,socket控制代碼就是乙個檔案描述符。fd_set集合可以通過一些巨集由人為來操作,比如清空集合:fd_zero(fd_set*),將乙個給定的檔案描述符加入集合之中fd_set(int, fd_set*),將乙個給定的檔案描述符從集合中刪除fd_clr(int, fd_set*),檢查集合中指定的檔案描述符是否可以讀寫fd_isset(int, fd_set*)。一會兒舉例說明。
第二:struct timeval是乙個大家常用的結構,用來代表時間值,有兩個成員,乙個是秒數,另乙個毫秒數。
具體解釋select的引數:
int maxfdp是乙個整數值,是指集合中所有檔案描述符的範圍,即所有檔案描述符的最大值加1,不能錯!在windows中這個引數值無所謂,可以設定不正確。
fd_set* readfds是指向fd_set結構的指標,這個集合中應該包括檔案描述符,我們是要監視這些檔案描述符的讀變化的,即我們關心是否可以從這些檔案中讀取資料了,如果這個集合中有乙個檔案可讀,select就會返回乙個大於0的值,表示有檔案可讀,如果沒有可讀的檔案,則根據timeout引數再判斷是否超時,若超出timeout的時間,select返回0,若發生錯誤返回負值。可以傳入null值,表示不關心任何檔案的讀變化。
fd_set* writefds是指向fd_set結構的指標,這個集合中應該包括檔案描述符,我們是要監視這些檔案描述符的寫變化的,即我們關心是否可以向這些檔案中寫入資料了,如果這個集合中有乙個檔案可寫,select就會返回乙個大於0的值,表示有檔案可寫,如果沒有可寫的檔案,則根據timeout再判斷是否超時,若超出timeout的時間,select返回0,若發生錯誤返回負值。可以傳入null值,表示不關心任何檔案的寫變化。
fe_set* errorfds同上面兩個引數的意圖,用來監視檔案錯誤異常。
struct timeval* timeout是select的超時時間,這個引數至關重要,它可以使select處於三種狀態。
第一:若將null以形參傳入,即不傳入時間結構,就是將select置於阻塞狀態,一定等到監視檔案描述符集合中某個檔案描述符發生變化為止;
第二:若將時間值設為0秒0毫秒,就變成乙個純粹的非阻塞函式,不管檔案描述符是否有變化,都立刻返回繼續執行,檔案無變化返回0,有變化返回乙個正值;
第三:timeout的值大於0,這就是等待的超時時間,即select在timeout時間內阻塞,超時時間之內有事件到來就返回了,否則在超時後不管怎樣一定返回,返回值同上述。
select函式返回值:
負值:select錯誤
正值:某些檔案可讀寫或出錯
0:等待超時,沒有可讀寫或錯誤的檔案
windows平台下解釋:
1,函式原型:
int select(int nfds, fd_set* readfds, fd_set* writefds, fd_set* exceptfds, const struct timeval* timeout);
2,引數:
nfds: 本引數忽略,僅起到相容作用,設為0即可;
readfds: (可選)指標,指向一組等待可讀性檢查的套介面;
writefds: (可選)指標,指向一組等待可寫性檢查的套介面;
exceptfds:(可選)指標,指向一組等待錯誤檢查的套介面;
timeout: 本函式最多等待時間,對阻塞操作則為null。
3,返回值:
(1)select()呼叫返回處於就緒狀態並且已經包含在fd_set結構中的描述字總數;
(2)如果超時則返回0;
(3)否則的話,返回socket_error錯誤,應用程式可通過wsagetlasterror()獲取相應錯誤**。
看下原始碼:
#ifndef fd_setsize
#define fd_setsize 64
#endif /* fd_setsize */typedef
struct
fd_set fd_set;
extern
int pascal far __wsafdisset(socket, fd_set far *);
#define fd_clr(fd, set) do /((fd_set far *)(set))->;fd_count--; /
break; /} /} /}
while(0
)#define fd_set(fd, set) do /} /
if (__i == ((fd_set far *)(set))->;fd_count) /} /}
while(0
)#define fd_zero(set) (((fd_set far *)(set))->;fd_count=0)
#define fd_isset(fd, set) __wsafdisset((socket)(fd), (fd_set far *)(set))typedef int32_t __fd_mask;
#define _nfdbits (sizeof(__fd_mask) * 8) /* 8 bits per byte */
#define __howmany(x,y) (((x)+((y)-1))/(y))#ifndef _fd_set
# define _fd_set
typedef
struct
__fd_set fd_set;
# ifndef _kernel
# ifdef __cplusplus
extern"c
",則select()立即返回,這可用於探詢所選套介面的狀態。如果處於這種狀態,則select()呼叫可認為是非阻塞的,且一切適用於非阻塞呼叫的假設都適用於它。
5,錯誤**:
wsanotinitialised:在使用此api之前應首先成功地呼叫wsastartup()。
wsaenetdown: windows套介面實現檢測到網路子系統失效。
wsaeinval: 超時時間值非法。
wsaeintr: 通過乙個wsacancelblockingcall()來取消乙個(阻塞的)呼叫。
wsaeinprogress: 乙個阻塞的windows套介面呼叫正在執行中。
wsaenotsock: 描述字集合中包含有非套介面的元素。
6,如何處理
上面在說明fd_setsize時,winsock2.h中定義fd_setsize的大小為64,這樣就對readfds、writefds、exceptfds的socket控制代碼數進行了限制。在實際應用中可以使用埠分組或者重新定義fd_setsize的方式進行解決。在stdafx.h最末行新增如下定義:
#define fd_setsize 1024 //socket控制代碼數
#define maximum_wait_objects 1024 //要等待的物件數
要注意的是我們還重定義了要另乙個巨集maximum_wait_objects,它表示要等待的物件數。重定義後,程式在現場執行正常。
socket通訊中的select應用簡要說明
用select的目的就是將多個io的讀寫阻塞的集中到乙個select上,即用select來檢測有沒有io可以讀寫,一旦發生至少乙個io可以讀寫 或者超時,返回0表示超時 則select就返回,返回值 0,然後用迴圈對select監測的所有io用fd isset來判斷,檢查是否是該io觸發的阻塞返回,...
socket通訊中select函式的使用和詳解
最近在寫乙個網路通訊函式,參考別人的 時對select 函式的使用存有疑惑,不太確定具體的使用方法,何時使用,以及其作用。在網上搜到一篇文章,覺得介紹的不錯,收藏學習。先自我總結一下。select函式的作用 select 在socket程式設計中還是比較重要的,可是對於初學socket的人來說都不太...
socket通訊中select函式的使用和詳解
最近在寫乙個網路通訊函式,參考別人的 時對select 函式的使用存有疑惑,不太確定具體的使用方法,何時使用,以及其作用。在網上搜到一篇文章,覺得介紹的不錯,收藏學習。先自我總結一下。select函式的作用 select 在socket程式設計中還是比較重要的,可是對於初學socket的人來說都不太...