一、基本原理
在linux编程基础中,select和poll的I/O多路转接复用模式是处理I/O复用的一个高效的方法。它可以具体设置程序中每一个所关心的文件描述符的条件、希望等待的时间等,从select()和poll()函数返回时,内核会通知用户已准备好的文件描述符的数量、已准备好的条件等。通过使用select()和poll()函数的返回结果,就可以调用相应的I/O 处理函数。在这种模型下,如果请求的I/O 操作阻塞,且它不是真正阻塞I/O,而是让其中的一个函数等待,在这期间,I/O 还能进行其他操作。
二、相关API接口
2.1 select函数
函数原型:
int select(int numfds, fd_set *readfds, fd_set *writefds,fd_set *exeptfds, struct timeval *timeout)
函数传入值:
numfds : 该参数值为需要监视的文件描述符的最大值加1
readfds : 由select()监视的读文件描述符集合
writefds :由select()监视的写文件描述符集合
exeptfds :由select()监视的异常处理文件描述符集合
timeout : 为NULL时,永远等待,直到捕捉到信号或文件描述符已准备好为止;为具体值时,struct timeval 类型的指针,若等待了timeout 时间还没有检测到任何文件描符准备好,就立即返回;为0时,从不等待,测试所有指定的描述符并立即返回
函数返回值:
大于0成功,返回准备好的文件描述符的数目;0时,超时;-1时,出错。
可以看到,select()函数根据希望进行的文件操作对文件描述符进行了分类处理,这里,对文件描述符的处理主要涉及4 个宏函数,这四个函数如下所示:
FD_ZERO(fd_set *set) 清除一个文件描述符集
FD_SET(int fd, fd_set *set) 将一个文件描述符加入文件描述符集中
FD_CLR(int fd, fd_set *set) 将一个文件描述符从文件描述符集中清除
FD_ISSET(int fd, fd_set *set) 如果文件描述符fd 为fd_set 集中的一个元素,则返回非零值,可以用于调用select()之后测试文件描述符集中的文件描述符是否有变化
一般来说,在使用select()函数之前,首先使用FD_ZERO()和FD_SET()来初始化文件描述符集,在使用了select()函数时,可循环使用FD_ISSET()来测试描述符集,在执行完对相关文件描述符的操作之后,使用FD_CLR()来清除描述符集。另外,select()函数中的timeout 是一个struct timeval 类型的指针,该结构体如下所示: