天天看點

select函數用法詳解

1. select函數

select的作用:

輪詢的方式,從多個檔案描述符中擷取狀态變化後的情況。

頭檔案

#include <sys/time.h>   //for struct timeval
#include <unistd.h>     //for select
           

函數原型

int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
           

參數解釋

  • nfds: 要檢測的檔案描述符的範圍,為檔案最大描述符+1
  • readfds: 包含所有可能因狀态變成可讀而觸發select函數傳回的檔案描述符
  • writefds: 包含所有可能因狀态變成可寫而觸發select函數傳回的檔案描述符
  • exceptfds: 包含所有可能因狀态發生異常而觸發select函數傳回的檔案描述符
void FD_CLR(int fd, fd_set *set);       //将fd在set中去除
void FD_SET(int fd, fd_set *set);       //将fd添加到set中
int  FD_ISSET(int fd, fd_set *set);     //檢查set中fd是否有狀态變化
void FD_ZERO(fd_set *set);              //清空整個set
           
  • timeout:
struct timeval {
     long    tv_sec;         /* seconds */
     long    tv_usec;        /* microseconds */
};
           

傳回值

當逾時或其中一個/多個檔案描述符發生變化時,此函數将傳回,傳回值:

  • -1: 函數執行出錯
  • 0: 逾時,并将時間結構體清空為0
  • 0: 狀态變化的檔案描述符的個數

基本執行個體

fd_set rdfds;   
struct timeval tv;
int rtn = 0;

FD_ZERO(&rdfds);     //清空fd_set
FD_SET(socket, &rdfds);  //将要檢測的socket描述符加入到fd_set集合中

tv.tv_sec = 2;
tv.tv_usec = 1000;      //設定逾時時間為2s+1ms

rtn = select(socket+1, &rdfds, NULL, NULL, &tv);

if(rtn < 0)
    perror("select");
else if(0 == rtn)
    printf("timeout\n");
else
{
    printf("rtn = %d\n", rtn);      //檢視有多少個檔案描述符發生了變化
    
    if(FD_ISSET(socket, &rdfds))    //判斷下這個socket是否狀态真的變成了可讀
    {
        recv(...);
    }
}
           

2. select做時間函數

void time_delay(int seconds)
{ 
    struct timeval tv;
    int err;
    
    tv.tv_sec = seconds;     //逾時時間
    tv.tv_usec = 0;
    
    do
    {
       err = select(0,NULL,NULL,NULL,&tv);  //将nfds設定為0即為逾時函數
    }while(err<0 && errno==EINTR);          //
}
           

3. 未完待續

繼續閱讀