• poll


    poll

    文件描述符突破1024, 修改配置文件
    监听和返回集合分离
    搜索范围变小

    int poll(struct pollfd *fds, nfds_t nfds, int timeout);
    如果不再监控某个文件描述符时, 可以把pollfd中, fd设置为-1, 破灵力不再监控此pollfd, 下次返回时, 把revents设置为0
    返回值: 返回满足条件事件数
    参数:
      fds: 数组的首地址
      ndfs: 数组中元素的个数
      timeout:
        -1: 阻塞等, #define INFTIM -1, linux中没有定义此宏
        0: 立即返回, 不阻塞
        >0: 等待指定毫秒数, 如当前系统时间精度不够毫秒, 向上取值

    struct pollfd {
       int   fd;         /* file descriptor */
       short events;     /* requested events */		// POLLIN, POLLOUT, POLLERR
       short revents;    /* returned events */		// 满足事件时, 操作系统填充事件
    };
    

    监听同一文件描述符的两种事件时, 可定义两个相同的文件描述符, event设置不同即可

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <netinet/in.h>
    #include <arpa/inet.h>
    #include <poll.h>
    #include <errno.h>
    #include <ctype.h>
    
    #include "wrap.h"
    
    #define MAXLINE 80
    #define SERV_PORT 8000
    #define OPEN_MAX 1024
    
    int main(int argc, char *argv[])
    {
        int i, j, maxi, listenfd, connfd, sockfd;
        int nready;                                 /*接收poll返回值, 记录满足监听事件的fd个数*/
        ssize_t n;
    
        char buf[MAXLINE], str[INET_ADDRSTRLEN];
        socklen_t clilen;
        struct pollfd client[OPEN_MAX];
        struct sockaddr_in cliaddr, servaddr;
    
        listenfd = Socket(AF_INET, SOCK_STREAM, 0);
    
        int opt = 1;
        setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    
        bzero(&servaddr, sizeof(servaddr));
        servaddr.sin_family = AF_INET;
        servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
        servaddr.sin_port = htons(SERV_PORT);
    
        Bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
        Listen(listenfd, 128);
    
        client[0].fd = listenfd;                    /* 要监听的第一个文件描述符 存入client[0]*/
        client[0].events = POLLIN;                  /* listenfd监听普通读事件 */
    
        for (i = 1; i < OPEN_MAX; i++)
            client[i].fd = -1;                      /* 用-1初始化client[]里剩下元素 0也是文件描述符,不能用 */
    
        maxi = 0;                                   /* client[]数组有效元素中最大元素下标 */
    
        for ( ; ; ) {
            nready = poll(client, maxi+1, -1);      /* 阻塞监听是否有客户端链接请求 */
    
            if (client[0].revents & POLLIN) {       /* listenfd有读事件就绪 */
    
                clilen = sizeof(cliaddr);
                connfd = Accept(listenfd, (struct sockaddr *)&cliaddr, &clilen);/* 接收客户端请求 Accept 不会阻塞 */
                printf("received from %s at PORT %d
    ",
                        inet_ntop(AF_INET, &cliaddr.sin_addr, str, sizeof(str)),
                        ntohs(cliaddr.sin_port));
    
                for (i = 1; i < OPEN_MAX; i++)
                    if (client[i].fd < 0) {
                        client[i].fd = connfd;      /* 找到client[]中空闲的位置,存放accept返回的connfd */
                        break;
                    }
    
                if (i == OPEN_MAX)                  /* 达到了最大客户端数 */
                    perr_exit("too many clients");
    
                client[i].events = POLLIN;          /* 设置刚刚返回的connfd,监控读事件 */
                if (i > maxi)
                    maxi = i;                       /* 更新client[]中最大元素下标 */
                if (--nready <= 0)
                    continue;                       /* 没有更多就绪事件时,继续回到poll阻塞 */
            }
    
            for (i = 1; i <= maxi; i++) {           /* 前面的if没满足,说明没有listenfd满足. 检测client[] 看是那个connfd就绪 */
                if ((sockfd = client[i].fd) < 0)
                    continue;
    
                if (client[i].revents & POLLIN) {
    
                    if ((n = Read(sockfd, buf, MAXLINE)) < 0) {
                        /* connection reset by client */
                        if (errno == ECONNRESET) {  /* 收到RST标志 */
                            printf("client[%d] aborted connection
    ", i);
                            Close(sockfd);
                            client[i].fd = -1;      /* poll中不监控该文件描述符,直接置为-1即可,不用像select中那样移除 */
                        } else
                            perr_exit("read error");
    
                    } else if (n == 0) {            /* 说明客户端先关闭链接 */
                        printf("client[%d] closed connection
    ", i);
                        Close(sockfd);
                        client[i].fd = -1;
                    } else {
                        for (j = 0; j < n; j++)
                            buf[j] = toupper(buf[j]);
                        Writen(sockfd, buf, n);
                    }
                    if (--nready <= 0)
                        break;
                }
            }
        }
        return 0;
    }
    
  • 相关阅读:
    Php开发学习---言简意赅,内含视频教程
    决策树剪枝的三种方法
    梯度弥散与梯度爆炸
    算法岗面试题积累一
    《转》从系统和代码实现角度解析TensorFlow的内部实现原理 | 深度
    算法复习-全排列的非递归和递归实现(含重复元素)
    xgboost使用细节
    pandas Series和dataframe
    python 字符词串和字符串的转换
    (转载)自然语言处理中的Attention Model:是什么及为什么
  • 原文地址:https://www.cnblogs.com/hesper/p/10739022.html
Copyright © 2020-2023  润新知