Linux-多路转接-select/poll

news2025/1/16 0:58:39

select/poll

  • 五种IO模型
    • 对IO的正确理解
    • 何为高效的IO
    • 阻塞IO
    • 非阻塞IO
      • 设置文件描述符为非阻塞模式
      • 非阻塞IO例子
    • 信号驱动IO
    • 异步IO
    • 多路转接
  • selct
    • 认识接口
      • select返回值
    • 基本使用
    • select使用特点
    • 缺点
  • poll
    • 认识接口
    • 对select的改善
    • 缺点

五种IO模型

对IO的正确理解

🚀IO不仅仅是数据的拷贝过程,而IO更准确的定义是:IO = 等待条件就绪 + 拷贝,就像在网络通信中客户端连接上服务器后但是一直不给服务器发送请求,这时对于服务器而言就无法读取,而是一直等待读取事件就绪。

何为高效的IO

🚀所谓高效的IO就是单位事件内,等待事件就绪的时间所占整个IO时间的比例越低,说明IO的效率越高。

阻塞IO

🚀阻塞IO是最常见的IO模型,所有文件默认都是阻塞IO。在资源就绪之前进程会一直处于等待状态,等待资源就绪后会对数据进行拷贝,然后返回。
在这里插入图片描述

非阻塞IO

🚀非阻塞IO,当进行IO时发现事件还未就绪就会直接返回,并且错误码被置为EGAIN 或者 EWOULDBLOCK。非阻塞IO是实现轮询的基础,通常程序员利用非阻塞IO来进行轮询,反复的尝试读写某个文件描述符,但是对CPU来说是较大的浪费,一般在特殊情况下使用。
在这里插入图片描述

在这里插入图片描述

设置文件描述符为非阻塞模式

void SetNonBlock(int fd) {
	int fl = fcntl(fd,F_GETFL);
    if(fl < 0) {
        std::cerr << "getfl faliled\n";
        return -1;
    }
    fcntl(0,F_SETFL,fl | O_NONBLOCK);
}

非阻塞IO例子

#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <cerrno>
#include <vector>
#include <functional>

using func_t = std::function<void (void)>;
std::vector<func_t> task_array;

void Log() {
    std::cout << "这是一个日志任务\n";
}
void Mysql() {
    std::cout << "这是一个数据库任务\n";
}
void Net() {
    std::cout << "这是一个网络任务\n";
}
void InitTaskArray() {
    task_array.push_back(Log);
    task_array.push_back(Mysql);
    task_array.push_back(Net);
}
void HandlerAllTask() {
    for(auto& func : task_array) {
        func();
    }
}

int main() {
    InitTaskArray();
    //1.设置对标准输入的非阻塞IO
    int fl = fcntl(0,F_GETFL);
    if(fl < 0) {
        std::cerr << "getfl faliled\n";
        return -1;
    }
    fcntl(0,F_SETFL,fl | O_NONBLOCK);
    //2.从标准输入读取数据 ---- ctl d 文件结尾
    while(true) {
        std::cout << ">>>";
        fflush(stdout);
        char buffer[256] {0};
        ssize_t n = read(0,buffer,sizeof(buffer) - 1);
        if(n > 0) {
            buffer[n - 1] = 0;
            std::cout << "读取到的内容: " <<buffer << std::endl;
        } else if(n == 0) {
            std::cout << "读到了文件结尾\n";
            break;
        } else if(n < 0) {
            //非阻塞IO时,读写条件没就绪也会出错返回但是错误码被设置成EAGAIN
            if(errno & EAGAIN) {
                //std::cout << "数据没有准备好\n";
                //数据没准备好
                sleep(1);
                HandlerAllTask();
                continue;
            } else if(errno & EINTR) {  //可能读取被信号打断
                continue;
            }
        } else {
            std::cerr << "读数据出错\n";
            return -1;
        }
    }
    return 0;
}

信号驱动IO

🚀信号驱动IO:就是内核将数据准备好的时候给应用进程发送SIGIO信号,通知进程进行IO操作,所以进程要对SIGIO信号进行捕捉。

在这里插入图片描述

异步IO

🚀异步IO与信号驱动IO很像,区别就是进程指发起IO事件,但真正的IO过程由内核完成,将数据拷贝成功后回像进程发送信号,通知进程。与信号驱动IO区别就是进程收到信号后不用自己调用read读取,而是收到信号时表明数据拷贝已经完成。

在这里插入图片描述

多路转接

🚀IO = 等待 + 数据拷贝,多路转接是将两个部分分开来做,由专门的系统调用来进行IO事件就绪的等待工作并且一个可以等待多个文件描述符,当事件就绪时会返回给应用进程,这时应用进程在去进行read/write进行数据拷贝,由于多路转接方案,一次可以等待多个文件描述符,所以在任意时刻事件准备就绪的概率更大,所以IO效率更高。

在这里插入图片描述

selct

🚀selct是Linux下实现多路转接的一个函数模型。

认识接口

int select(int nfds, fd_set *readfds, fd_set *writefds,
                  fd_set *exceptfds, struct timeval *timeout);

void FD_CLR(int fd, fd_set *set);
int  FD_ISSET(int fd, fd_set *set);
void FD_SET(int fd, fd_set *set);
void FD_ZERO(fd_set *set);

  • nfds:值为等待的多个文件描述符中最大的那个值加一。

  • readfds:关心读事件的文件描述符集合

    🚀fd_set:是一个位图结构,其每个比特位的位置就表示是哪个文件描述符,内容设置为1,就表示关心这个文件描述符的读/写/异常事件。

  • writefds:关心写事件的文件描述符集合

  • exceptfds:关心异常事件的文件描述符集合

  • timeout:设置select多长事件轮询一次
    在这里插入图片描述
    🚀如果timeval中两个字段设置为0,表示非阻塞轮询方式。如果设定了某个事件,表示以设置的时间间隔进行轮询。将该字段设为nullptr表示阻塞式等待。

🚀FD_CLR:不再关心fd的事件
🚀FD_ISSET:判断位图中是否包含fd
🚀FD_SET:将fd设置进位图,让select关心其事件
🚀FD_ZERO:清空整个位图

注意:select的后四个参数都是输入输出型参数,对于三个位图结构:调用select时表明用户告诉内核想让内核关心哪些文件描述的哪些事件,select返回时,表明让内核关心的哪些文件描述符的哪些事件已经就绪。也就是说在每次返回时都会对我们调用前设置的位图结构更改,所以每次调用前都要重新设置位图结构,从而也就导致第一个参数也要跟着改变。
struct timeval* timeout:当被设置为某个时间间隔后,当事件就绪select返回时,其值会被设置为剩余时间。

select返回值

🚀1.大于0,返回的是包含在三个就绪事件中的文件描述符的数量。
🚀2.等于0,timeout字段被设置,表示经过timeout时间没有事件就绪,返回一次。
🚀3.小于0,select出错

基本使用

🚀下面的代码中由于没有指定应用层协议,所以数据读取和发送是不正确的,在后面分享的epoll中,会制定应用层协议,进行正确的读取和发送。

代码仓库: select

select使用特点

🚀可监控的文件描述符数量取决于sizeof(fd_set),不同的系统可能不同,fd_set作为一个数据类型那么其大小就是确定的,sizeof(fd_set)*8表示select能监控的最多文件描述符数量。
🚀将fd加入到select的监控集合后,还有另开辟一个数组来保存设置进select监控集中的fd。
🚀另开辟数组保存fd的原因:1.select返回后,array作为源数据进行FD_ISSET判断,判断某个fd是否在返回的fd_set集合中。2.select返回后会把以前加入的但并无事件发生的fd清空,则每次开始select前都要重新从array取得fd逐一加入(FD_ZERO最先),扫描array的同时取得fd最大值maxfd,用于select的第一个参数。

缺点

🚀每次调用select, 都需要手动设置fd集合, 从接口使用角度来说也非常不便
🚀每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时比较大
🚀同时每次调用select都需要在内核遍历传递进来的所有fd,这个开销在fd很多时也比较大
🚀在应用层当select返回时,也要一次遍历array中的所有fd,判断fd相关事件是否就绪
🚀select支持的文件描述符数量太小,相比于一个进程能够管理的文件描述符数量。

poll

认识接口

int poll(struct pollfd *fds, nfds_t nfds, int timeout);
struct pollfd {
    int   fd;         /* file descriptor */
  	short events;     /* requested events */
    short revents;    /* returned events */
   };

🚀第一个参数是要用户传递一个数组进去,数组的每个元素为struct pollfd结构体,内容包括fd,告诉内核要关心的事件events,内核告诉用户哪些事件已经就绪revents。

events的常用选项:
在这里插入图片描述

🚀第二个参数为数组中有效元素的个数。
🚀第三个参数是超时时间:
大于0:经过timeout时间的等待没有时间就绪,就返回。
等于0:没有时间就绪直接返回。
-1:阻塞方式等待。

对select的改善

🚀1.相对于select而言,poll可以关心的文件描述符数量没有上限。
🚀2.poll的pollfd参数做到了输入输出参数分离,意味着不用用户在每次调用poll函数之前都要重新设置要让内核关心哪些fd上的哪些事件。

缺点

🚀1.和select一样,每次都需要将pollfd结构由用户态拷贝到内核态,当数量较多时开销比较大。
🚀2.每次调用poll时,在内核中都要对传递进来的fd进行遍历,在fd很多时开销较大。
🚀3.每次poll返回时,用户也要对pollfd数组进行遍历,fd数量很多时,开销比较大。
🚀4.同时连接的大量客户端在任意时刻可能只有很少的处于就绪状态,因此随着监视的文件描述符数量的增多,其效率会有所下降。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1036649.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL学习笔记9

MySQL数据表中的数据类型&#xff1a; 在考虑数据类型、长度、标度和精度时&#xff0c;一定要仔细地进行短期和长远的规划&#xff0c;另外&#xff0c;公司制度和希望用户用什么方式访问数据也是要考虑的因素。开发人员应该了解数据的本质&#xff0c;以及数据在数据库里是如…

Sentinel故障转移及实现原理

Sentinel故障转移及实现原理 一、哨兵模式的基本工作流程二、判断实例下线三、选举新主库四、哨兵模式弊端五、哨兵集群判断实例下线六、哨兵集群判断实例下线详细工作过程七、哨兵集群的通信八、哨兵和客户端的通信九、总结 一、哨兵模式的基本工作流程 redis在运行时会开启一…

vue基础知识十五:说说你对slot的理解?slot使用场景有哪些?

一、slot是什么 在HTML中 slot 元素 &#xff0c;作为 Web Components 技术套件的一部分&#xff0c;是Web组件内的一个占位符 该占位符可以在后期使用自己的标记语言填充 举个栗子 <template id"element-details-template"><slot name"element-na…

(2022|ECCV,图像分割,VQ-SEG,AR Transformer)Make-A-Scene:利用人类先验进行基于场景的文本到图像生成

Make-A-Scene: Scene-Based Text-to-Image Generation with Human Priors 公众号&#xff1a;EDPJ&#xff08;添加 VX&#xff1a;CV_EDPJ 或直接进 Q 交流群&#xff1a;922230617 获取资料&#xff09; 目录 0. 摘要 1. 简介 2.相关工作 2.1. 图像生成 2.2. 图像标…

2023年汉字小达人区级自由报名明天开赛,3个新问题和往年真题练一练

明天9月25日&#xff0c;备受关注的2023年第十届上海小学生汉字小达人区级自由报名的比赛就要开始了&#xff0c;最近还是有几个“小迷糊”家长刚听说这个活动&#xff0c;问了几个问题&#xff0c;我觉得挺有普遍性的&#xff0c;所以再次给大家回答一下&#xff0c;希望能够帮…

redis漏洞修复:CVE-2022-35977、CVE-2023-22458、CVE-2023-28856

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、漏洞内容二、现状三、更新redis下载镜像停止已有的容器启动新的容器 四、更新后的版本1. 查看日志2. 查看版本 总结 前言 漏扫发现机器上的redis版本有点低…

图像识别-YOLO V8安装部署-window-CPU-Pycharm

前言 安装过程中发现&#xff0c;YOLO V8一直在更新&#xff0c;现在是2023-9-20的版本&#xff0c;已经和1月份刚发布的不一样了。 eg: 目录已经变了&#xff0c;旧版预测:在ultralytics/yolo/v8/下detect 新版&#xff1a;ultralytics/models/yolo/detect/predict.py 1.安…

九、多项式朴素贝叶斯算法(Multinomial NB,Multinomial Naive Bayes)(有监督学习)

Multinomial Naive Bayes&#xff1a;用于多项式模型的Naive Bayes分类器 一、算法思路 多项式Naive Bayes分类器适用于离散特征分类&#xff08;如文本分类中的字数&#xff09; 多叉分布通常需要整数特征计数 不过&#xff0c;在实际应用中&#xff0c;分数计数&#xff08…

LeetCode刷题

一 【移除元素】 原题链接&#xff1a;27. 移除元素 - 力扣&#xff08;LeetCode&#xff09; 给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值等于 val 的元素&#xff0c;并返回移除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须仅使用…

SLAM从入门到精通(机器人建模和仿真环境)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 很多同学学了ros&#xff0c;以为把publish、subscribe、消息、服务这些接口学好了就行。其实这是很大的误区。因为这些通信机制只是帮我们了解ros…

bean加载properties文件(spring)

1.开启Context命名空间 复制一下第二行的信息,然后粘贴一下,把粘贴过来的所有beans替换成context 2.使用context命名空间,加载指定的properties文件 3,使用#{}占位符读取加载的属性值 实际实例(在配置文件xml中写) 补充: 不加载系统属性的解释: 如果你在配置文件中配置的变量…

MyBatis友人帐之缓存

一、概述 1.1简介 什么是缓存 [ Cache ]&#xff1f; 存在内存中的临时数据。 将用户经常查询的数据放在缓存&#xff08;内存&#xff09;中&#xff0c;用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询&#xff0c;从缓存中查询&#xff0c;从而提高查询效率&…

【SoC基础】硬件起源之晶体管的诞生

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…

Go语言入门篇

目录 一、基础数据类型 1.1 变量的定义方式 1.2 用%T输出变量的类型 二、复合数据类型 2.1 数组 2.1.2、数组的遍历 2.1.3 数组传参 2.2. 切片slice 2.2.1. 初始化切片 2.2.2. append向切片中追加元素 2.2.3. 切片的截取 2.3. map 2.3.1. map初始化 2.3.2. 添加和…

CSS3有哪些新特性

CSS3 引入了许多新特性&#xff0c;以增强样式设计和页面布局的能力&#xff0c;提供更多的视觉效果和交互性。以下是一些 CSS3 中的新特性&#xff1a; 圆角边框&#xff08;Border Radius&#xff09;&#xff1a;圆角的边框&#xff0c;而不是传统的方形边框。 <!DOCTY…

GLTF编辑器告诉你凹凸贴图的作用

什么是凹凸贴图 凹凸贴图&#xff08;Bump Mapping&#xff09;是一种计算机图形学中的技术&#xff0c;用于在表面上模拟微小的凹凸形状&#xff0c;从而增加了物体的细节和真实感。它可以在不改变物体几何形状的情况下&#xff0c;通过修改光照的反应&#xff0c;使表面看起来…

华为云云耀云服务器L实例评测 | WebVR性能测试

随着最近几年Web技术的快速发展&#xff0c;Web3D和WebVR在网页端效果越来越好。 本文讲述如何在云耀云服务器L实例上部署WebVR服务器&#xff0c;服务器代码里添加一个3D模型&#xff0c;然后再使用本地浏览器作为客户端来查看模型加载和交互情况&#xff0c;并体验WebVR。 …

猴赛雷 ! 上次我见过这么厉害的安全测试实战演练还是上次!

01、概念介绍 1.1 xss XSS 攻击通常指的是通过利用网页开发时留下的漏洞&#xff0c;通过巧妙的方法注入恶意指令代码到网页&#xff0c;使用户加载并执行攻击者恶意制造的网页程序。这些恶意网页程序通常是 JavaScript&#xff0c;但实际上也可以包括 Java、 VBScript、Acti…

Python绘制二元函数图像

1 问题 利用python来解决绘制二元函数图像的问题。 2 方法 用文字描述解题思路&#xff0c;可配合一些图形以便更好的阐述。解决问题的步骤采用如下方式&#xff1a; 注意下述步骤全部使用(1)格式。 需要调用两个第三方库;Matplotlib、numpy编写代码绘制f(x,y)(sin(x)*sin(y))/…

云上亚运:所使用的高新技术,你知道吗?

作者简介&#xff1a;一名云计算网络运维人员、每天分享网络与运维的技术与干货。 公众号&#xff1a;网络豆云计算学堂 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a; 网络豆的主页​​​​​ 目录 前言 一.什么是云上亚运会 二.为什么要使用云…