目录
int poll(struct pollfd* fds , nfds_t nfds , int timeout);
int epoll_ctl(int epfd , int op , int fd , struct epoll_event *event)
一、select
#include<sys/select.h>
int select(int nfds , fs_set* readfds , fd_set* writefds , fd_set* exceptfds , struct timeval* timeout);
- nfds参数指定被监听的文件描述符的总数
- readfds、writefds和exceptfds参数分别指向可读、可写和异常等事件对应的文件描述符集合
下代码段为fd_set描述
#include<typesizes.h> #define __FD_SETSIZE 1024 #include<sys/select.h> #define FD_SETSIZE __FD_SETSIZE typedef long int __fd_mask; #undef __NFDBITS #define __NFDBITS (8 * (int)sizeof (__fd_mask)) typedef struct { #ifdef __USE_XOPEN __fd_mask fds_bits[__FD_SETSIZE / __NFDBITS]; #else __fd_mask __fds_bits[__FD_SETSIZE / NFDBITS]; #define __FDS_BITS(set) ((set)->__fds_bits) #endif }fd_set
由上定义可见,fd_set结构体仅包含一个整数数组,该数组的每个元素的每一位标记一个文件描述符。
由于位操作过于繁琐,可以用下面一系列宏来访问fd_set结构体中的位
#include<sys/select.h>
FD_ZERO(fd_set *fdset); /*清除fdset的所有位*/
FD_SET(int fd,fd_set *fdset); /*设置fdset的位fd*/
FD_CLR(int fd,fd_set *fdset); /*清除fdset的位fd*/
int FD_ISSET(int fd,fd_set *fdset); /*测试fdset的位fd是否被设置*/
*
select成功时返回就绪(可读、可写和异常)文件描述符的总数。如果在超时时间内没有任何文件描述符就绪,select将返回0。select失败时返回-1并设置errno。如果在select等待期间,程序接收到信号,则select立即返回-1,并设置errno为EINTR。
通常用FD_ISSET来判断文件描述符是否就绪
二、文件描述符就绪条件
哪些情况下文件描述符可以被认为是可读、可写或者出现异常,对于select的使用非常关键。在网络编程中,下列情况下socket可读。
- socket内核接收缓存区中的字节数大于或等于其低水位标记SO_RCVLOWAT。此时我们可以无阻塞地读该socket,并且读操作返回的字节数大于0.
- socket通信的对方关闭连接,此时对该socket的读操作将返回0.
- 监听socket上有新的连接请求。
- socket上有未处理的错误。此时我们可以使用getsockopt来读取