select 函数详解

Unix系统下解释:

头文件: #include  <sys/time.h>

函数原型:int select(int maxfdp, fd_set* readfds, fd_set* writefds, fd_set* errorfds, struct timeval* timeout);

select函数的主要用途是什么?select函数是一个监控函数,用来监控插口的状态!插口就是指的是文件描述符,Unix / Linux一切皆文件。

select函数能够监控插口状态,如下表:

具体解释select的参数:

先解释第五个参数(最后一个参数):struct timeval  *timeout是select的超时时间,这个参数至关重要,它可以使select处于三种状态。

1、若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,一定等到监视文件描述符集合中某个文件描述符发生变化为止;

2、若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,不管文件描述符是否有变化,都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;

3、timeout的值大于0,这就是等待的超时时间,即select在timeout时间内阻塞,超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回,返回值同上述。

第一个参数: int maxfdp是一个整数值,是指集合中所有文件描述符的范围,即所有文件描述符的最大值加1,不能错!

第二个参数: 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同上面两个参数的意图,用来监视文件错误异常。这个通常是:OOB同步标记未处理。

 

再说两个结构体:

第一:struct fd_set可以理解为一个集合,这个集合中存放的是文件描述符(file descriptor),即文件句柄,这可以是我们所说的普通意义的文件,当然Unix下任何设备、管道、FIFO等都是文件形式,全部包括在内,所以毫无疑问,一个 socket 就是一个文件,socket句柄就是一个文件描述符。

fd_set 集合需要用一些宏来操作

FD_ZERO(fd_set*);       清空集合,相当于memset

FD_SET(int, fd_set*);     将一个给定的文件描述符加入集合之中

FD_CLR(int,   fd_set*);   将一个给定的文件描述符从集合中删除

FD_ISSET(int, fd_set*);  检查集合中指定的文件描述符是否可以读写

    第二:struct timeval 是一个大家常用的结构,用来代表时间值,有两个成员,一个是秒数,另一个毫秒数。

struct timeval{      
        long tv_sec;   /*秒 */
        long tv_usec;  /*微秒 */
    }

 

select函数返回值:

负值:select错误

0:等待超时,没有可读写或错误的文件

正值:有文件可读,或可写,或可读可写

 

在TCP网络编程中,常见的判断链路状态的代码如下:

一次判断

int ret = 0;
int l_errno = 0;
fd_set rfd, efd;
struct timeval tv;

FD_ZERO(&rfd);
FD_ZERO(&efd);
FD_SET(fd, &efd);
FD_SET(fd, &rfd);

tv.tv_sec = 3;
tv.tv_usec = 0;

ret = select(fd+1, &rfd, NULL, &efd, &tv);    //这里不关心写,只关心读和异常状态
l_errno = errno;

if(ret < 0)
{
    DEBUG_MSG(" TCP link error %d   errno %d \n", ret, l_errno);
    return -1;
}

 

循环判断 

int ret = 0;
int l_errno = 0;
fd_set rfd, efd;
struct timeval tv;

while(1)
{    
    FD_ZERO(&rfd);
    FD_ZERO(&efd);
    FD_SET(fd, &efd);
    FD_SET(fd, &rfd);

    tv.tv_sec = 2;
    tv.tv_usec = 0;

    ret = select(fd+1, &rfd, NULL, &efd, &tv);    //这里不关心写,只关心读和异常状态
    l_errno = errno;

    switch(ret)
    {
        case -1:    //异常,需要在下面进行返回判断处理
            break;

        case 0:     // 可能是超时
            break;

        default:    //正常状态,判断到有可读的文件,需要在下面判断返回
            break;
    }

    //其他处理,确保跳出while循环
    /*   */
}

 

  • 5
    点赞
  • 68
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值