socket通訊中select函式的使用和解釋

2021-09-07 04:54:11 字數 4437 閱讀 3311

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的人來說都不太...