select系统调用是用来让我们的程序监视多个文件描述符(file descrīptor)的状态变化的。程序会停在select这里等待,直到被监视的文件描述符有某一个或多个发生了状态改变。
select()的机制中提供一fd_set的数据结构,实际上是一long类型的数组,每一个数组元素都能与一打开的文件描述符(不管是Socket描述符,还是其他文件或命名管道或设备描述符)建立联系,建立联系的工作由程序员完成, 当调用select()时,由内核根据IO状态修改fd_set的内容,由此来通知执行了select()的进程哪一Socket或文件可读,
系统提供了select函数实实现多路复用输入/输出模型。select函数原型如下:
参数nfds(有点地方也叫maxfd)是需要监视的最大的文件描述符值+1;
readfds, writefds, exceptfds分别对应于需要检测的可读文件描述符的集合,可写文件描述符的集合及异常文件描述符的集合。
定义的这三个参数都是描述符的集合,第一个rfds是用来保存这样的描述符的:当描述符的状态变成可读的时系统就会告诉select函数返回,第二个wfds是指有描述符状态变成可写的时系统就会告诉select函数返回,第三个参数efds是特殊情况,即描述符上有特殊情况发生时系统会告诉select函数返回。
函数的最后一个参数timeout是一个超时时间值,其类型是struct timeval *,即一个指向struct timeval结构的变量的指针,所以我们在程序里要声明一个struct timeval tv;然后把变量tv的地址&tv传递给select函数。struct timeval结构如下:
另外关于fd_set类型的变量,还有一组标准的宏定义来处理此类变量:
fd_set
FD_ZERO(&fdset):清空fdset与所有文件描述符的联系。//将fdset清零,使集合中不含任何fd。
FD_SET(int fd, &fdset):建立文件描述符fd与fdset的联系。//将fd加入fdset集合
FD_CLR(int fd, &fdset):清除文件描述符fd与fdset的联系。//将fd从fdset集合中删除
FD_ISSET(int fd, &fdset):检查fd_set联系的文件描述符fd是否可读写,>0表示可读写。//测试fd是否在集合fdset中
(关于fd_set及相关宏的定义见/usr/include/sys/types.h)
下面以一个输入为例来说明:
int fd1, fd2;
fd1 = socket(...);
fd2 = open(“/dev/tyS0”,O_RDWR);
FD_ZERO(&rfds);
FD_SET(fd1, &rfds);
FD_SET(fd2, &rfds);
int maxfd = 0;
maxfd = (fd1>fd2)?(fd1+1):(fd2+1);
ret = select(maxfd, &rfds, NULL, NULL, &tv);
这样就可以使用一个开关语句(switch语句)来判断到底是哪一个输入源在输入数据。具体判断如下:
switch(ret)
{
}