select
作用:select 函数就是这样一种常见的 I/O 多路复用技术,使用 select 函数,通知内核挂起进程,当一个或多个 I/O 事件发生后,控制权返还给应用程序,由应用程序进行 I/O 事件的处理。
参数:
int select(int maxfd, fd_set *readset, fd_set *writeset,
fd_set *exceptset, const struct timeval *timeout);
返回:若有就绪描述符则为其数目,若超时则为0,若出错则为-1
在这个函数中,maxfd 表示的是待测试的描述符基数,它的值是待测试的最大描述符加 1。Linux系统中select支持的maxfd的最大值为1024,比如现在的 select 待测试的描述符集合是{0,1,4},那么 maxfd 就是 5。紧接着的是三个描述符集合,分别是读描述符集合 readset、写描述符集合 writeset 和异常描述符集合 exceptset,这三个分别通知内核,在哪些描述符上检测数据可以读,可以写和有异常发生。三个描述符集合中的每一个都可以设置成空,这样就表示不需要内核进行相关的检测。
描述符集合函数:
设置描述符集合相关函数,其中的每个元素都是二进制中的0或者1:a[maxfd-1], ..., a[1], a[0]
void FD_ZERO(fd_set *fdset); //所有元素设为0
void FD_SET(int fd, fd_set *fdset); //a[fd]设为1
void FD_CLR(int fd, fd_set *fdset); //a[fd]设为0
int FD_ISSET(int fd, fd_set *fdset); //a[fd]是否为1
时间参数:
最后一个参数是 timeval 结构体时间:
struct timeval {
long tv_sec; /* seconds */
long tv_usec; /* microseconds */
};
这个参数设置成不同的值,会有不同的可能:
1.可能是设置成空 (NULL),表示如果没有 I/O 事件发生,则 select 一直等待下去。
2.可能是设置一个非零的值,这个表示等待固定的一段时间后从 select 阻塞调用中返回,
3.可能是将 tv_sec 和 tv_usec 都设置成 0,表示根本不等待,检测完毕立即返回。
注意:
●描述符基数是当前最大描述符 +1
●select调用返回时会改变监听的描述符集合,所以要循环监听的话需要在开始重置描述符集合
●select函数对fd有1024的限制,不是0-1023的fd会导致未定义的行为
使用:
fd_set readmask;
fd_set allreads;
FD_ZERO(&allreads);
FD_SET(0, &allreads);
FD_SET(socket_fd, &allreads);
for (;;) {
readmask = allreads; //readmask被改变了,要重置
int rc = select(socket_fd + 1, &readmask, NULL, NULL, NULL);
if (rc <= 0) {
printf("select failed");
exit(0);
}
if (FD_ISSET(socket_fd, &readmask)) //如果socket_fd有I/O事件
...
套接字描述符就绪条件
●读:内核通知我们套接字有数据可以读了,使用 read 函数不会阻塞。
1.套接字接收缓冲区有数据可以读,如果我们使用 read 函数去执行读操作,肯定不会被阻塞,而是会直接读到这部分数据。
2.是对方发送了 FIN,使用 read 函数执行读操作,不会被阻塞,直接返回 0。
3.是针对一个监听套接字而言的&#x