zoukankan      html  css  js  c++  java
  • Linux非阻塞IO(八)使用epoll重新实现非阻塞的回射服务器

    本文无太多内容,主要是几个前面提到过的注意点:

    一是epoll的fd需要重新装填。我们将tcp_connection_t的指针保存在数组中,所以我们以这个数组为依据,重新装填fd的监听事件

    //重新装填epoll内fd的监听事件
            int i;
            for(i = 0; i < EVENTS_SIZE; ++i)
            {
                if(connsets[i] != NULL)
                {
                    int fd = i; //fd
                    tcp_connection_t *pt = connsets[i]; //tcp conn
                    uint32_t event = 0;
                    if(buffer_is_readable(&pt->buffer_))
                        event |= kWriteEvent;
                    if(buffer_is_writeable(&pt->buffer_))
                        event |= kReadEvent;
                    //重置监听事件
                    epoll_mod_fd(epollfd, fd, event);
                }
            }

    二是,建立连接时,需要做的工作是:

    1.新建tcp_connection_t结构,初始化

    2.将fd加入epoll,不监听任何事件

    3.将tcp_connection_t的指针加入数组。

    代码如下:

    //建立连接
    int peerfd = accept4(listenfd, NULL, NULL, SOCK_NONBLOCK | SOCK_CLOEXEC);
    if(peerfd == -1)
          ERR_EXIT("accept4");
    //新建tcp连接
    tcp_connection_t *pt = (tcp_connection_t*)malloc(sizeof(tcp_connection_t));
    buffer_init(&pt->buffer_);
    //将该tcp连接放入connsets
    connsets[peerfd] = pt;
    epoll_add_fd(epollfd, peerfd, 0);

    连接关闭时需要:

    //close
    epoll_del_fd(epollfd, fd);
    close(fd);
    free(pt);
    connsets[fd] = NULL;

    还有一点:前面我们记录fd和connsets的关系,采用的是数组下标的方式,其实我们还可以将指针存入epoll的data中,其中:

    typedef union epoll_data {
       void        *ptr;
       int          fd;
       uint32_t     u32;
       uint64_t     u64;
    } epoll_data_t;
    
    struct epoll_event {
       uint32_t     events;      /* Epoll events */
       epoll_data_t data;        /* User data variable */
    };

    我们对于data这个联合体,不再使用fd,而是使用ptr,指向一个tcp_connection_t的指针。不过我们需要将fd存储在tcp_connection_t数据结构中。

    这里为了简便起见,仍采用以前的方法,读者可以自行尝试。

    完整的代码如下:

    #define _GNU_SOURCE             /* See feature_test_macros(7) */
    #include <sys/socket.h>
    #include "sysutil.h"
    #include "buffer.h"
    #include <assert.h>
    #include <sys/epoll.h>
    
    #define EVENTS_SIZE 1024
    
    typedef struct{
        buffer_t buffer_;
    } tcp_connection_t; //表示一条TCP连接
    
    tcp_connection_t *connsets[EVENTS_SIZE]; //提供从fd到TCP连接的映射
    
    int main(int argc, char const *argv[])
    {
        //获取监听fd
        int listenfd = tcp_server("localhost", 9981);
        //将监听fd设置为非阻塞
        activate_nonblock(listenfd);
    
        //初始化connsets
        int ix;
        for(ix = 0; ix < EVENTS_SIZE; ++ix)
        {
            connsets[ix] = NULL;
        }
    
    
        //初始化epoll
        int epollfd = epoll_create1(0);
        epoll_add_fd(epollfd, listenfd, kReadEvent);
        struct epoll_event events[1024];
    
    
        while(1)
        {
            //重新装填epoll内fd的监听事件
            int i;
            for(i = 0; i < EVENTS_SIZE; ++i)
            {
                if(connsets[i] != NULL)
                {
                    int fd = i; //fd
                    tcp_connection_t *pt = connsets[i]; //tcp conn
                    uint32_t event = 0;
                    if(buffer_is_readable(&pt->buffer_))
                        event |= kWriteEvent;
                    if(buffer_is_writeable(&pt->buffer_))
                        event |= kReadEvent;
                    //重置监听事件
                    epoll_mod_fd(epollfd, fd, event);
                }
            }
    
            //epoll监听fd
            int nready = epoll_wait(epollfd, events, 1024, 5000);
            if(nready == -1)
                ERR_EXIT("epoll wait");
            else if(nready == 0)
            {
                printf("epoll timeout.
    ");
                continue;
            }
    
            //处理fd
            for(i = 0; i < nready; ++i)
            {
                int fd = events[i].data.fd;
                uint32_t revents = events[i].events;
                if(fd == listenfd) //处理listen fd
                {
                    if(revents & kReadREvent)
                    {
                        //建立连接
                        int peerfd = accept4(listenfd, NULL, NULL, SOCK_NONBLOCK | SOCK_CLOEXEC);
                        if(peerfd == -1)
                            ERR_EXIT("accept4");
                        //新建tcp连接
                        tcp_connection_t *pt = (tcp_connection_t*)malloc(sizeof(tcp_connection_t));
                        buffer_init(&pt->buffer_);
                        //将该tcp连接放入connsets
                        connsets[peerfd] = pt;
                        epoll_add_fd(epollfd, peerfd, 0);
                    }
                }
                else //处理普通客户的fd
                {
                    //取出指针
                    tcp_connection_t *pt = connsets[fd];
                    assert(pt != NULL);
                    if(revents & kReadREvent)
                    {
                        if(buffer_read(&pt->buffer_, fd) == 0)
                        {
                            //close
                            epoll_del_fd(epollfd, fd);
                            close(fd);
                            free(pt);
                            connsets[fd] = NULL;
                            continue; //继续下一次循环
                        } 
                    }
    
                    if(revents & kWriteREvent)
                    {
                        buffer_write(&pt->buffer_, fd);
                    }
                }
            }
        }
    
        close(listenfd);
    
        return 0;
    }

    下文使用epoll的ET模式。

  • 相关阅读:
    【Linux】【jenkins】自动化部署一 安装jenkins及Jenkins工作目录迁移
    【Linux】【docker】docker私服安装
    【Linux】【docker】docker及docker-compose安装
    【Linux】【tomcat】tomcat8.5安装
    【Linux】【jdk】jdk8.0安装
    【Linux】【mysql】mysql8.0开启远程访问及常见问题
    【Linux】记录一个yum update和upgrade的区别
    【Linux】【gitlab】gitlab安装、备份、恢复、升级、内存消耗问题
    Python序列——列表
    Python序列——元组
  • 原文地址:https://www.cnblogs.com/inevermore/p/4055077.html
Copyright © 2011-2022 走看看