select
概述
将监听的文件描述符分为三组,每一组监听不同的需要进行的IO操作(读、写、异常)。当select返回时,无差别轮询所有流,找出能读出数据,或者写入数据的流,每组文件描述符会被select过滤,只留下可以进行对应IO操作的文件描述符。
有数量限制:受单个进程能打开的文件描述符数量受 FD_SIZE(默认1024) 限制,修改这个宏需要重新编译内核。
也看到有另一种说法如下(说法来源于《网络编程实战》C10K问题章节):
-
使用
ulimit -n
查看单个进程能打开的文件描述符数量(默认 1024) -
用 root 权限修改
/etc/sysctl.conf
文件,使得系统可以支持10000个描述符上限fs.file-max = 10000 net.ipv4.ip_conntrack_max = 10000 net.ipv4.netfilter.ip_conntrack_max = 10000
函数声明
int select(int maxfd, fd_set *readset, fd_set *writeset, fd_set *exceptset, const struct timeval *timeout);
//返回:若有就绪描述符则为其数目,若超时则为0,若出错则为-1
maxfd
表示的是待测试的描述符基数,它的值是待测试的最大描述符加1(因为是从 0 开始计数的)。比如现在的select
待测试的描述符集合是{0,1,4},那么maxfd
就是5。
紧接着的是三个描述符集合,分别是读描述符集合readset
、写描述符集合writeset
和异常描述符集合exceptset
,这三个分别通知内核,在哪些描述符上检测数据可以读,可以写和有异常发生。
使用方法
- 循环从 socket_fd 中筛选出已就绪的感兴趣内容
- 就绪之后调用对应的方法进行相应的操作
int main(int argc, char **argv) {
if (argc != 2) {
error(1, 0, "usage: select01 ");
}
int socket_fd = tcp_client(argv[1], SERV_PORT);
char recv_line[MAXLINE], send_line[MAXLINE];
int n;
fd_set readmask;
fd_set allreads;
FD_ZERO(&allreads); //初始化
FD_SET(0, &allreads); // 0 是标准输入输出
FD_SET(socket_fd, &allreads); // 套接字描述符 3
for (;;) {
readmask = allreads;
int rc = select(socket_fd + 1, &readmask, NULL, NULL, NULL);
if (rc <= 0) {
error(1, errno, "select failed");
}
if (FD_ISSET(socket_fd, &readmask)) {
n = read(socket_fd, recv_line, MAXLINE);
if (n < 0) {
error(1, errno, "read error");
} else if (n == 0) {
error(1, 0, "server terminated ");
}
recv_line[n] = 0;
fputs(recv_line, stdout);
fputs("", stdout);
}
if (FD_ISSET(STDIN_FILENO, &readmask)) {
if (fgets(send_line, MAXLINE, stdin) != NULL) {
int i = strlen(send_line);
if (send_line[i - 1] == '') {
send_line[i - 1]