首页 > 其他分享 >61 多路转接poll

61 多路转接poll

时间:2024-08-04 11:27:27浏览次数:10  
标签:转接 int pos 61 fds fd poll event

poll函数的接口

#include <poll.h>
int poll(struct pollfd fds, nfds_t nfds, int timeout);
// pollfd结构
struct pollfd {
int fd; /
file descriptor /
short events; /
requested events /
short revents; /
returned events */
};

参数说明
fds是一个poll函数监听的结构列表,每一个元素中,包含了三部分内容:文件描述符,监听的事件集合,返回的事件集合
nfds表示数组fds的长度
timeout表示poll函数的超时时间,单位是毫秒(ms),每隔多少秒检测一次,0非阻塞,-1阻塞
events是用户需要关心的事件,用户告诉内核,renents是关心事件就绪情况,内核告诉用户
events和revents的取值:

事件描述是否可作为输入是否可作为输出
POLLIN数据(包括普通数据和优先数据)可读
POLLRDNORM普通数据可读
POLLRDBAND优先级带数据可读(linux不支持)
POLLPRI高优先级数据可读,比如tcp带外数据
POLLOUT数据(包括普通数据和优先数据)可写
POLLWRNORM普通数据可写
POLLWRBAND优先级带数据可写
POLLERR错误
POLLHUP挂起。比如管道的写端被关闭后,读端描述符上将收到POLLHUP事件
POLLNVAL文件描述符没有打开

返回结果
返回值小于0,表示出错
返回值等于0,表示poll函数等待超时
返回值大于0,表示poll由于监听的文件描述符就绪而返回

socket就绪条件

和select一样

poll示例

PollServer类维护liten套接字和一个pollfd的数组,保存所有增加的套接字
初始化函数初始化套接字设置listen状态
运行函数先将listen套接字设置到数组第一个位置,设置关注事件读,timeout延时3秒,死循环调用poll函数获取状态,大于0说明有事件处理,调用事件分配函数
分配函数遍历数组取就绪事件fd,如果是监听套接字就将获得的fd插入到数组中,设置监听的事件。如果不是就读取缓冲区内容
PollServer

#include <poll.h>
#include "Socket.hpp"
#include "log.hpp"

static const uint16_t defaultport = 8000;
static const int fd_max = 64;
int defaultfd = -1;
int non_event = 0;

class PollServer
{
public:
    PollServer()
    {
        for (int i = 0; i < fd_max; i++)
        {
            _event_fds[i].fd = defaultfd;
            _event_fds[i].events = non_event;
            _event_fds[i].revents = non_event;
        }
    }

    bool Init()
    {
        _listensocket.Socket();
        int opt = 1;
        setsockopt(_listensocket._sockfd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt));
        _listensocket.Bind(defaultport);
        _listensocket.Listen();

        return true;
    }

    void Accepter()
    {
        cout << "get a new link" << endl;
        std::string clientip;
        uint16_t clinetport = 0;
        int sock = _listensocket.Accept(&clientip, &clinetport);
        if (sock < 0)
        {
            return;
        }
        lg.logmessage(info, "accept success,%s:%d.%d", clientip.c_str(), clinetport, sock);
        // 添加sock
        int pos = 1;
        for (; pos < fd_max; pos++)
        {
            if (_event_fds[pos].fd != defaultfd)
            {
                continue;
            }
            else
            {
                break;
            }
        }

        if (pos == fd_max)
        {
            lg.logmessage(warning, "server is full,close %d", sock);
            close(sock);
            // 可以扩容
        }
        else
        {
            _event_fds[pos].fd = sock;
            _event_fds[pos].events = POLLIN;
            _event_fds[pos].revents = non_event;
            // 打印查看fdary
            PrintFd();
        }
    }

    void Recver(int fd, int pos)
    {
        char buff[1024];
        ssize_t n = read(fd, buff, sizeof(buff) - 1);
        if (n > 0)
        {
            buff[n] = 0;
            cout << "get message:" << buff << endl;
        }
        else if (n == 0)
        {
            lg.logmessage(info, "client quit, me too, fd:", fd);
            close(fd);
            _event_fds[pos].fd = defaultfd; // 这里本质是移除
        }
        else
        {
            lg.logmessage(warning, "recv error,fd:", fd);
            close(fd);
            _event_fds[pos].fd = defaultfd; // 这里本质是移除
        }
    }

    void Dispatcher()
    {
        // 遍历找出就绪描述符
        for (int i = 0; i < fd_max; i++)
        {
            int fd = _event_fds[i].fd;
            if (fd == defaultfd)
                continue;

            if (_event_fds[i].revents & POLLIN)
            {
                // 监听就绪
                if (fd == _listensocket.Fd())
                {
                    Accepter();
                }
                else // 读就绪
                {
                    Recver(fd, i);
                }
            }
        }
    }

    void Start()
    {
        // listen套接字放入第一个,并设置读事件
        _event_fds[0].fd = _listensocket.Fd();
        _event_fds[0].events = POLLIN;
        int timeout = 3000;  // 3s
        for (;;)
        {
            // poll函数
            int n = poll(_event_fds, fd_max, timeout);
            // 就绪不处理,会一直通知,这时读取不会阻塞
            switch (n)
            {
            case 0:
                cout << "timeout..." << endl;
                break;
            case -1:
                cerr << "poll error" << endl;
                break;
            default:
                // 有事件就绪,todo
                Dispatcher();
                break;
            }
        }
    }

    void PrintFd()
    {
        cout << "online fd list: ";
        for (int i = 0; i < fd_max; i++)
        {
            if (_event_fds[i].fd == defaultfd)
                continue;
            cout << _event_fds[i].fd << " ";
        }
        cout << endl;
    }

    ~PollServer()
    {
        _listensocket.Close();
    }

private:
    Sock _listensocket;
    struct pollfd _event_fds[fd_max];  // 用户维护的数组
};


PollServer.cc

#include <memory>
#include "PollServer.hpp"

int main()
{
    std::unique_ptr<PollServer> svr(new PollServer);
    svr->Init();
    svr->Start();
    return 0;
}

poll的优点

不同于select使用三个位图来表示fdset的方式,poll使用一个pollfd的指针实现
pollfd结构包含了要监视的ecent和发生的evnet,不再使用select“参数-值”传递的方式,接口使用比select更方便
poll并没有最大数量限制(但是数量过大后性能也是会下降)

poll的缺点

poll中监听的文件描述符数目增多时
和select函数一样,poll返回后,需要轮询pollfd来获取就绪的描述符
每次调用poll都需要把大量的pollfd结构从用户态拷贝到内核中
同时连接的大量客户端在一时刻可能只有很少的处于就绪状态,因此随着监视的描述符数量的增长,其效率会线性下降

标签:转接,int,pos,61,fds,fd,poll,event
From: https://blog.csdn.net/qq_43422358/article/details/140774746

相关文章

  • 63 epoll服务器 (ET模式)
    基于LT模式修改,并加入前面的应用层计算器,实现稍完整的服务器功能1.修改tcp_socket.hpp,新增非阻塞读和非阻塞写接口2.对于accept返回的new_sock加上EPOLLET这样的选项注意:此代码暂时未考虑listen_sockET的情况,如果将listen_sock设为ET,则需要非阻塞轮询的方式accept,否则会......
  • M2 Pro 本地docker部署apollo
    M2Pro本地docker部署apollo1.环境说明2.部署前准备3.部署整体步骤说明4.部署流程4.1部署MySQL(如果本地已部署,直接跳到4.2步骤)4.2执行Apollo的SQL初始化脚本4.3部署apollo-configservice拉取apollo-configservice镜像启动容器端口映射更改带来的配置变动4.4......
  • 使用epoll编写TCP服务器示例
    #include<stdio.h>#include<stdlib.h>#include<string.h>#include<errno.h>#include<netinet/in.h>#include<sys/socket.h>#include<arpa/inet.h>#include<sys/epoll.h>#include<unistd.h>#include......
  • 【代码随想录训练营第42期 Day17打卡 二叉树Part5-LeetCode 654.最大二叉树 617.合并
    目录一、做题心得二、题目与题解题目一:654.最大二叉树题目链接题解:递归题目二:617.合并二叉树题目链接题解:递归(前序遍历)题目三:617.合并二叉树题目链接题解:BFS层序遍历 题目四:98.验证二叉搜索树题目链接题解:递归(中序遍历)三、小结一、做题心得今天是代码随想......
  • 代码随想录day17 || 654 最大二叉树,617 合并二叉树,700 二叉搜索树搜索,98 验证二叉搜索
    645最大二叉树funcconstructMaximumBinaryTree(nums[]int)*TreeNode{ //思路,算法思路基本等同于通过中序前序构造二叉树 //1,取最大值作为根节点 //2,切割数组 //3,递归左右子树 iflen(nums)==0{ returnnil } //切割数组取最大值 max,left,right:=......
  • 打卡信奥刷题(487)用Scratch图形化工具信奥P1161[普及组/提高] 开灯
    开灯题目描述在一条无限长的路上,有一排无限长的路灯,编号为1,2,3,......
  • 洛谷P3161 [CQOI2012] 模拟工厂 贪心策略的思考
    P3161[CQOI2012]模拟工厂传送门:模拟工厂问题描述:初始的生产力和商品分别为1和0。每一个时刻可以选择两个动作:生产力+1或者生产生产力数量的商品。现在有很多个订单,每个订单有三个部分:时间t,需要多少商品p,可以获得的价值v。现在需要决定各个时刻的动作选择,以及订单是否接取,以期......
  • CodeForces 1619D New Year's Problem
    题目链接:CodeForces1619D【NewYear'sProblem】思路    可以因为最多只能逛n-1个商店,当n-1大于等于m的时候,所有朋友都能取最大值,否则至少有两个人要选择相同的商店,所以依次枚举两个人选择同一个商店,其他人选择喜悦值最大的商店。代码#include<cstddef>#incl......
  • Luogu P1613 跑路 题解 [ 蓝 ] [ 倍增 ] [ Floyd 最短路 ] [ 状压 dp ]
    跑路:绝佳倍增好题,思路是化\(2^k\)为\(1\),倍增起预处理作用。最近不知道是撞了什么运,前一脚看的是绿题,写完之后交一发,发现直接被lxl升蓝了,血赚。思路:Floyd首先观察到每次走\(2^k\)的代价为\(1\),我们可以预处理出每次走\(2^i\)能到哪些点。但为了让这题的代码更好实......
  • 洛谷 B3612 【深进1.例1】求区间和
    "这道题也太水了吧,模拟就行了!""数据范围...""好像不行呀""呜呜~~TLE!"献上暴力代码!#include<bits/stdc++.h>usingnamespacestd;constintN=1e5+5;intn,a[N],m;signedmain(){ios::sync_with_stdio(0);cin.tie(0);......