Reactor反应堆模式

reactor 反应堆模式

运行结果:

补充:

1、通信套接字

cpp 复制代码
    static void HandleRequest(Connection* conn)
    {
        string s;
        while(DeLenHeader(conn->_inbuffer, &s))
        {
            Request req;
            req.DeSerialize(s);

            int result;
            switch(req._op)
            {
            case '+':
                result= req._x+ req._y;
                break;
            case '-':
                result= req._x- req._y;
                break;
            case '/':
                result= req._x/ req._y;
                break;
            case '*':
                result= req._x* req._y;
                break;
            }
            
            Response rsp(result, 0);

            string sen= rsp.Serialize();
            EnLenHeader(sen);

            conn->_outbuffer+= sen;
        }

        if(!conn->_outbuffer.empty())
        {
            conn->_server->ModifyEvents(conn->_fd, EPOLLIN | EPOLLOUT| EPOLLET);
        }
    }

    static void HandleRead(Connection* conn)
    {
        std::string& inbuffer= conn->_inbuffer;
        char buffer[1024];

        while(1)
        {
            int n= read(conn->_fd, buffer, sizeof(buffer)-1);
            if(n> 0)
            {
                buffer[n]= 0;
                inbuffer+= buffer;
            }
            else if(n== 0)
            {
                conn->_server->DelConnection(conn);
                conn->_server->DelEvents(conn->_fd);
                close(conn->_fd);
                delete conn;
                return;
            }
            else if(n== -1 && errno== EAGAIN)
            {
                break;
            }
        }

        HandleRequest(conn);
    }

    static void HandleWrite(Connection* conn)
    {
        while(1)
        {
            int n= write(conn->_fd, conn->_outbuffer.c_str(), conn->_outbuffer.size());//参数传的是要写的数据,返回值是实际写入发送缓冲区的数据字节数

            if(n> 0)
            {
                conn->_outbuffer.erase(0, n);
                if(conn->_outbuffer.empty())
                {
                    break;
                }
            }
            else if(n==-1 && errno== EAGAIN)
            {
                break;
            }
        }

        //走到这里,可能是数据全部写进了发送缓冲区,也可能是发送缓冲区满了
        //如果是因为全写进了发送缓冲区,那么就不需要监控可写事件了,如果是发送缓冲区满了,那就还得监控,等发送缓冲区的数据由内核发送出去后,会执行ep_poll_callback,将struct epitem加进就绪队列里
        if(conn->_outbuffer.empty())
        {
            conn->_server->ModifyEvents(conn->_fd, EPOLLIN |EPOLLET);
        }
    }

如果我们往发送缓冲区中一次写很多数据,那么很可能最后只能写入一部分,然后write会返回写入发送缓冲区的数据的字节数,如果写是非阻塞的,那么当发送缓冲区满了的时候,write会直接返回-1,设置errno为EAGAIN

上面的代码中,当对接受缓冲区中的数据接收并处理过之后,会将结果数据保存在另一个缓冲区中,然后启动可写事件监控,当结果数据没了之后,那就关闭可写事件监控

2、epoll_ctl

epoll_ctl有三种操作

  • EPOLL_CTL_ADD :用于添加新的文件描述符,如果已有对应文件描述符则添加失败

  • EPOLL_CTL_MOD :用于修改已监控文件描述符的事件(覆盖式)

  • EPOLL_CTL_DEL :用于移除文件描述符的监控

相关推荐
树℡独6 小时前
ns-3仿真之应用层(五)
服务器·网络·tcp/ip·ns3
小小管写大大码7 小时前
如何让vscode变得更智能?vscode接入claude实现自动编程
运维·ide·vscode·自动化·编辑器·ai编程·腾讯云ai代码助手
嵩山小老虎7 小时前
Windows 10/11 安装 WSL2 并配置 VSCode 开发环境(C 语言 / Linux API 适用)
linux·windows·vscode
Fleshy数模7 小时前
CentOS7 安装配置 MySQL5.7 完整教程(本地虚拟机学习版)
linux·mysql·centos
zhang133830890757 小时前
CG-09H 超声波风速风向传感器 加热型 ABS材质 重量轻 没有机械部件
大数据·运维·网络·人工智能·自动化
a41324478 小时前
ubuntu 25 安装vllm
linux·服务器·ubuntu·vllm
Configure-Handler8 小时前
buildroot System configuration
java·服务器·数据库
津津有味道8 小时前
易语言TCP服务端接收刷卡数据并向客户端读卡器发送指令
服务器·网络协议·tcp·易语言
Fᴏʀ ʏ꯭ᴏ꯭ᴜ꯭.9 小时前
Keepalived VIP迁移邮件告警配置指南
运维·服务器·笔记
物联网软硬件开发-轨物科技9 小时前
【轨物洞见】告别“被动维修”!预测性运维如何重塑老旧电站的资产价值?
运维·人工智能