函数原型

  1. #include <sys/select.h>
  2. int select(int nfds, fd_set *readfds, fd_set *writefds,
  3. fd_set *exceptfds, struct timeval *timeout);

参数

nfds

指定被监听的文件描述符的总数。它通常被设置为select()监听的最大文件描述符加1(文件描述符从0计数)

readfds、writefds、exceptfds

分别指向可读、可写和异常事件对应的文件描述符集合。这三个参数都是fd_set类型。
这三个参数是值-结果参数,在select()返回的的时候,将把就绪的fd更新到对应的集合中。

timeout

设置select()超时时间。

  • NULL: select()将无限等待
  • 非NULL: select()等待相应秒数,若无就绪则超时返回
    • 特例非NULL,但其值为0秒: select()检查完fd集合后立即返回

这是个值-结果参数,select()成功返回时,内核将修改它的值为剩余的时间。
但当该调用失败的时候,它返回的值是不确定的。该结构体(timeval)提供微秒级的分辨率。

返回值

  • 成功时返回就绪(可读、可写和异常)文件描述符的总数。
  • 如果在超时时间内没有任何文件描述符就绪,select()将返回0。
  • 失败时返回-1,并设置errno。

如果在select()等待期间,程序接收到信号,则select()立即返回-1,并设置errno为EINTR


fd_set

一个结构体名(被typedef命名,所以使用时不加struct)。实际上里面只包含一个整型(long int)数组。
该数组的每一位标记一个文件描述符。可以通过以下函数宏来访问fd_set结构体中的位。

返回值 函数宏 描述

void|FD_ZERO(fd_set set)|清除set所有的位
void|FD_CLR(int fd, fd_set
set)|清除set的为fd
void|FD_SET(int fd, fd_set set)|设置set的位fd
int |FD_ISSET(int fd, fd_set
set)|测试set的位fd是否被设置

在使用fd_set类型的变量的时候,一定要用FD_ZERO初始化