select、poll、epoll 底层探讨

select

概述

将监听的文件描述符分为三组,每一组监听不同的需要进行的IO操作(读、写、异常)。当select返回时,无差别轮询所有流,找出能读出数据,或者写入数据的流,每组文件描述符会被select过滤,只留下可以进行对应IO操作的文件描述符。

有数量限制:受单个进程能打开的文件描述符数量受 FD_SIZE(默认1024) 限制,修改这个宏需要重新编译内核。

也看到有另一种说法如下(说法来源于《网络编程实战》C10K问题章节):

  1. 使用 ulimit -n 查看单个进程能打开的文件描述符数量(默认 1024)

  2. 用 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,这三个分别通知内核,在哪些描述符上检测数据可以读,可以写和有异常发生。

使用方法

  1. 循环从 socket_fd 中筛选出已就绪的感兴趣内容
  2. 就绪之后调用对应的方法进行相应的操作
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] 
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值