select函数原型:
int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
1. 参数较多,不利于理解:
select有五个参数,每个参数都要单独维护,操作相对复杂。
2. 文件描述符数量限制:
select 使用 fd_set 数据结构来存储文件描述符,这个数据结构的大小是固定的。在一些系统上,这个大小被限制为 FD_SETSIZE(通常是 1024),这意味着 select 最多只能监视 1024 个文件描述符。
3. 效率问题:
每次调用 select 都需要将所有的文件描述符从用户空间复制到内核空间,这可能会导致性能问题,特别是当需要监视大量的文件描述符时。select 的效率可能不如其他的多路 I/O 复用技术,如 epoll。
4. 不可扩展性:
当需要监视的文件描述符数量增加时,select 的性能可能会下降,因为它需要遍历整个fd_set。
5. 不支持超时精度:
select 的超时参数是一个 struct timeval 结构,它的精度只能到微秒级别。对于某些应用,这可能不够精确。
6. 事件驱动的局限性:
select 是一个事件驱动的机制,但它只能监视读、写和错误三种类型的事件。对于其他类型的事件(如新连接的到来),select 是无法处理的。
7. 不支持非阻塞 I/O:
select 通常与阻塞 I/O 一起使用,如果需要使用非阻塞 I/O,你可能需要额外的代码来处理。