C/C++:Linux select 1024 文件描述符限制

2024-03-25 16:48

本文主要是介绍C/C++:Linux select 1024 文件描述符限制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C/C++:Linux select 1024 文件描述符限制

通常来说,Linux下select调用要求文件描述符的值小于1024,也就是说,fd set中的每个文件描述符的值域为:[0,1023]。

如果超过,Linux下select调用会发生什么?

环境:

[test1280@localhost ~]$ uname -a
Linux localhost.localdomain 3.10.0-327.el7.x86_64 #1 SMP Thu Nov 19 22:10:57 UTC 2015 x86_64 x86_64 x86_64 GNU/Linux

方法:

通过dup2生成指定数值的文件描述符,并将其与标准输入关联,select的读集合包含且只包含指定数值文件描述符。

代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>int main(int argc, char *argv[])
{if (argc != 2){fprintf(stderr, "usage: %s fd\n", argv[0]);exit(1);}int fd = atoi(argv[1]);if (fd < 3) // <0 0 1 2{fprintf(stderr, "fd domain: (2, fd_max]\n");exit(1);}fprintf(stdout, "fd size: %d\n", FD_SETSIZE);// 创建指定文件描述符,与标准输入关联if (dup2(0, fd) == fd){fprintf(stdout, "dup2 ok! new fd: %d\n", fd);}else{fprintf(stderr, "dup2 error: %d(%s)\n", errno, strerror(errno));exit(1);}// 设为非阻塞long flags = fcntl(fd, F_GETFL);	fcntl(fd, F_SETFL, flags|O_NONBLOCK);fprintf(stdout, "will into main loop...\n");while (1){fd_set readSet;FD_ZERO(&readSet);FD_SET(fd, &readSet);struct timeval tv;tv.tv_sec = 3;tv.tv_usec = 0;int res = select(fd + 1, &readSet, NULL, NULL, &tv);if (res == -1) {fprintf(stderr, "select error: %d(%s)\n", errno, strerror(errno));exit(1);} else if (res == 0) {fprintf(stdout, "timeout...\n");} else {static char buff[1024];int count = read(fd, buff, sizeof(buff));fprintf(stdout, "select return %d, read count %d\n", res, count);}}exit(0);
}

编译:

[test1280@localhost ~]$ gcc -o main main.c -g -Wall

运行:

case 1(正常态):

[test1280@localhost ~]$ ./main 10
fd size: 1024
dup2 ok! new fd: 10
will into main loop...
timeout...
timeout...
test1280timeout...select return 1, read count 9
timeout...
^C

case 2(异常态):

[test1280@localhost ~]$ ./main 1280
fd size: 1024
dup2 ok! new fd: 1280
will into main loop...
select error: 9(Bad file descriptor)

可读、可写、异常文件描述符集合包含大于1024的文件描述符时,执行select将返回EBADF错误。

但是,并不总是这样的!

在另一个程序中,将一个大于1024的FD加入到select可读监听集合执行select时,发生:

此FD没有输入数据,不可读,select并没有等待指定的时间返回0,而是立刻返回一个正数指代存在可读FD。

我没有能在以上的代码中复现这个问题。(以上代码执行select发现非法FD,直接返回EBADF错误)

在网络上别人的博客中,曾出现过段错误的异常:

https://my.oschina.net/u/1780368/blog/491863

man 2 select:

POSIX allows an implementation to define an upper limit, advertised
via the constant FD_SETSIZE, on the range of file descriptors that
can be specified in a file descriptor set. The Linux kernel imposes
no fixed limit, but the glibc implementation makes fd_set a fixed-
size type, with FD_SETSIZE defined as 1024, and the FD_*() macros
operating according to that limit. To monitor file descriptors
greater than 1023, use poll(2) instead.
According to POSIX, select() should check all specified file
descriptors in the three file descriptor sets, up to the limit
nfds-1. However, the current implementation ignores any file
descriptor in these sets that is greater than the maximum file
descriptor number that the process currently has open. According to
POSIX, any such file descriptor that is specified in one of the sets
should result in the error EBADF.

An fd_set is a fixed size buffer. Executing FD_CLR() or FD_SET()
with a value of fd that is negative or is equal to or larger than
FD_SETSIZE will result in undefined behavior.

Although there are some ways to increase this limit (because of you’re using Linux, everything is possible), select() call can only listen 1024 file descriptor at the same time and all of the file descriptor number must be less than 1024.

总结:

1.select要求三个入参被监听集合的FD(取值)不超过1024;由此可推2、4:

2.select要求三个入参被监听集合的FD(数量)不超过1024;

3.若超过1024,执行select可能发生段错误,可能发生立刻返回不阻塞(以及错误的返回值结果),等等。

4.即使select监听集合中只有一个FD,如果FD值大于1024,同样有问题。

参考:

1.https://linux-tips.com/t/is-it-possible-to-listen-file-descriptor-greater-than-1024-with-select/45
2.https://my.oschina.net/u/1780368/blog/491863
3.http://man7.org/linux/man-pages/man2/select.2.html
4.http://bbs.chinaunix.net/thread-1791112-1-1.html
5.https://blog.codingnow.com/2014/02/select_bug.html

这篇关于C/C++:Linux select 1024 文件描述符限制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/845618

相关文章

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

Linux内核定时器使用及说明

《Linux内核定时器使用及说明》文章详细介绍了Linux内核定时器的特性、核心数据结构、时间相关转换函数以及操作API,通过示例展示了如何编写和使用定时器,包括按键消抖的应用... 目录1.linux内核定时器特征2.Linux内核定时器核心数据结构3.Linux内核时间相关转换函数4.Linux内核定时

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

Linux镜像文件制作方式

《Linux镜像文件制作方式》本文介绍了Linux镜像文件制作的过程,包括确定磁盘空间布局、制作空白镜像文件、分区与格式化、复制引导分区和其他分区... 目录1.确定磁盘空间布局2.制作空白镜像文件3.分区与格式化1) 分区2) 格式化4.复制引导分区5.复制其它分区1) 挂载2) 复制bootfs分区3)

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias