【多路IO复用】select

news/2024/11/13 4:11:41/

select:

1.select:当被监听的 fd(文件描述符)就绪后会返回,但是我们无法知道具体是哪些 fd 就绪了,只能遍历所有的 fd。通常来说某一时刻,就绪的 fd 并不会很多,但是使用 select 必须要遍历所有的 fd,这就造成了一定程度上的性能损失。select 最多可监听的 fd 是有限制的,32位操作系统默认1024个,64位默认2048

select函数的API:

select函数的API
#include <sys/select.h>
 /* According to earlier standards */
       #include <sys/time.h>
       #include <sys/types.h>
       #include <unistd.h>

       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  : 最大文件描述符+1
    readfds : 需要监听的读的文件描述符存放集合
    writefds :需要监听的写的文件描述符存放集合   NULL
    exceptfds : 需要监听的异常的文件描述符存放集合  NULL
    timeout: 多长时间监听一次   固定的时间,限时等待   NULL 永久监听
    struct timeval {
               long    tv_sec;         /* seconds */ 秒
               long    tv_usec;        /* microseconds */微妙
           };

  返回值: 返回的是变化的文件描述符的个数



注意: 变化的文件描述符会存在监听的集合中,未变化的文件描述符会从集合中删除

select实现原理:

应用层中父进程通过内核的selsect监听文件描述符缓冲区的变化,内核就会返回给父进程

以fd_set为例,每次都要从用户态拷贝至内核态,同时还要在内核态进行循环遍历,然后把有事件的响应的文件描述符fd_set返回,又要从内核态拷贝至用户态。用户态拿到这个有事件的文件描述符返回,还要针对返回的描述符进行遍历,才能知道哪个文件描述符对应的Socket可写可读,总共经历了两次遍历,两次拷贝,所以说为什么Select在文件描述符比较多的情况,效率为什么是低下的原因。

select 的优缺点:

优点: 跨平台

缺点:

文件描述符1024的限制 由于 FD_SETSIZE的限制

只是返回变化的文件描述符的个数,具体哪个那个变化需要遍历

每次都需要将需要监听的文件描述集合由应用层符拷贝到内核

大量并发,少了活跃,select效率低

 


http://www.ppmy.cn/news/77078.html

相关文章

docker容器的时区更改

如果你希望Docker容器使用特定的时区&#xff0c;你有几种方法可以实现这个目标&#xff1a; 在运行Docker容器时设置环境变量&#xff1a;你可以在使用 docker run 命令启动容器时&#xff0c;通过 -e 参数设置一个环境变量 TZ。例如&#xff0c;如果你希望容器使用北京时间&a…

过滤表filter达式cql相互转化

CQL----sql字符串表达方式 Filter---过滤器表达方式 filter: 1,cqltofilter Filter filter CQL.toFilter(FilterCQLSample.LESS_FILTER_SAMPLE);---非中文 2,用FilterFactory filterFactory CommonFactoryFinder.getFilterFactory(null);创建 tostring: Filter…

10-Docker发布微服务

文章目录 搭建SpringBoot项目发布微服务项目到Docker容器 搭建SpringBoot项目 搭建一个简单的SpringBoot项目&#xff1a; 创建maven工程&#xff0c;pom为&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://m…

【OpenCv • c++】几何检测 —— 霍夫变换 | 霍夫直线检测 | 霍夫线变化

&#x1f680; 个人简介&#xff1a;CSDN「博客新星」TOP 10 &#xff0c; C/C 领域新星创作者&#x1f49f; 作 者&#xff1a;锡兰_CC ❣️&#x1f4dd; 专 栏&#xff1a;【OpenCV • c】计算机视觉&#x1f308; 若有帮助&#xff0c;还请关注➕点赞➕收藏&#xff…

Flutter实现PopupMenu(弹出设置菜单)

PopupMenuButton简介 PopupMenuButton是一个用于创建弹出菜单的小部件。它通常与IconButton或其他触发菜单显示的小部件一起使用。当用户点击触发按钮时&#xff0c;PopupMenuButton会在屏幕上方或下方弹出一个菜单&#xff0c;显示一组选项供用户选择。 PopupMenuButton常用…

Unknown system variable ‘query_cache_size‘

报错如下 The error occurred while executing a query ### Cause: org.springframework.jdbc.CannotGetJdbcConnectionException: Failed to obtain JDBC Connection; nested exception is java.sql.SQLException: Unknown system variable query_cache_size at org.myba…

2172. 最大公约数

Powered by:NEFU AB-IN Link 文章目录 2172. 最大公约数题意思路代码 2022年第十三届决赛真题 2172. 最大公约数 题意 给定一个数组, 每次操作可以选择数组中任意两个相邻的元素 x , y x, yx,y 并将其 中的一个元素替换为 gcd ⁡ ( x , y ) \operatorname{gcd}(x, y)gcd(x,y),…

go手写Redis(8)之数据库核心层及指令实现

数据库核心层 前面实现完了处理器的逻辑&#xff0c;现在到了核心的数据层实现了&#xff0c;核心的数据库主要是来执行用户发送的指令并且进行数据存储 1. Database 数据层的顶级接口定义&#xff0c;在 interface/database/database.go 文件中定义&#xff0c; 其中定义了…