一。select函数总结

阻塞方式block,就是进程或是线程执行到这些函数时必须等待某个事件的发生,如果事件没有发生,进程或线程就被阻塞,函数不能立即返回。使用Select就可以完成非阻塞non-block,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回值的不同来反映函数的执行情况,如果事件发生则与阻塞方式相同,若事件没有发生则返回一个代码来告知事件未发生,而进程或线程继续执行,所以效率较高。select能够监视我们需要监视的文件描述符的变化情况。

(一)首先说明两个结构体:
1:struct fd_set一个存放文件描述符(file descriptor),即文件句柄的聚合,实际上是一long类型的数组,
每一个数组元素都能与一打开的文件句柄(不管是Socket句柄,还是其他文件或命名管道或设备句柄)建立联系,建立联系的工作由程序员完成;

FD_ZERO(fd_set *fdset):清空fdset与所有文件句柄的联系。
FD_SET(int fd, fd_set *fdset):建立文件句柄fd与fdset的联系。
FD_CLR(int fd, fd_set *fdset):清除文件句柄fd与fdset的联系。
FD_ISSET(int fd, fdset *fdset):检查fdset联系的文件句柄fd是否可读写,>0表示可读写。

2:struct timeval用来代表时间值,有两个成员,一个是秒数tv_sec,另一个是毫秒数tv_usec。

(二)下面介绍select()函数原型:
1:int select(int nfds, fd_set *rdfds, fd_set *wtfds, fd_set *exfds, struct timeval *timeout)
2:ndfs:select中监视的文件句柄数,一般设为要监视的文件中的最大文件号加一。
3:rdfds:select()监视的可读文件句柄集合,当rdfds映象的文件句柄状态变成可读时系统告诉select函数返回。
这个集合中有一个文件可读,select就会返回一个大于0的值,表示有文件可读,
如果没有可读的文件,则根据timeout参数再判断是否超时,
若超出timeout的时间,select返回0,若发生错误返回负值,
可以传入NULL值,表示不关心任何文件的读变化;
4:wtfds: select()监视的可写文件句柄集合,当wtfds映象的文件句柄状态变成可写时系统告诉select函数返回。
如果这个集合中有一个文件可写,select就会返回一个大于0的值,表示有文件可写,
如果没有可写的文件,则根据timeout参数再判断是否超时,
若超出timeout的时间,select返回0,若发生错误返回负值,
可以传入NULL值,表示不关心任何文件的写变化。
5:exfds:select()监视的异常文件句柄集合,当exfds映象的文件句柄上有特殊情况发生时系统会告诉select函数返回。
6:timeout:select()的超时结束时间。
这个参数它使select处于三种状态,
第一,若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,
一定等到监视文件描述符集合中某个文件描述符发生变化为止;
第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,不管文件描述符是否有变化,
都立刻返回继续执行,文件无变化返回0,有变化返回一个正值;
第三,timeout的值大于0,这就是等待的超时时间,即select在timeout时间内阻塞,
超时时间之内有事件到来就返回了,否则在超时后不管怎样一定返回,返回值同上述。

7:返回值:负值:select错误
0:等待超时,没有可读写或错误的文件
正值:某些文件可读可写或出错

(三)下面是一个有三个套接字句柄的例子

int sa, sb, sc;
sa = socket(...);
connect(sa,...);
sb = socket(...);
connect(sb,...);
sc = socket(...);
connect(sc,...);
FD_SET(sa, &rdfds);/* 分别把3个句柄加入读监视集合里去 */
FD_SET(sb, &rdfds);
FD_SET(sc, &rdfds);
int maxfd = 0;
if(sa > maxfd) maxfd = sa;/* 获取3个句柄的最大值 */
if(sb > maxfd) maxfd = sb;
if(sc > maxfd) maxfd = sc;
struct timeval tv;
tv.tv_sec   = ... ;
tv.tv_usec = ...;
ret = select(maxfd + 1, &rdfds, NULL, NULL, &tv); /* 注意是最大值加1 */
if(ret < 0)
{
    perror("select");  /* select函数出错 */
}
else if(ret == 0)
{
    printf("超时\n"); /* 在设定的tv时间内,socket的状态没有发生变化 */
}
else
{
    printf("ret=%d\n", ret);
    if(FD_ISSET(sa, &rdfds)) /* 先判断一下sa这个被监视的句柄是否真的变成可读的了 */
    {
        recv(...);  /* 读取socket句柄里的数据 */
    }
    ......
}

二。select句柄数限制

connect中使用了select模型,有如下地方需要注意:
我们提供的server api中有很多地方用到了select,特别是在等超时的时候,
例如:

fd_set recv_fds;
int iNum= 0;
if (m_iSocket <0) return -1;
FD_ZERO( &recv_fds );
FD_SET( m_iSocket, &recv_fds );
iNum= select( m_iSocket+1, &recv_fds, NULL, NULL, timeout );
return iNum;

这段代码对于cgi,或者简单的逻辑server不会有问题。但是对于多线程或者复杂的server可能会导致server core掉。

原因是select默认只支持1024个句柄,每个句柄采用和1024个bit对应的关系,如果fd的值超过1024,那么就会溢出。
也就是说,如果上面代码的m_iSocket>1024,那么后面的select就会溢出,即使只监听一个句柄也会溢出。奇怪的是select也不会报错。
而对于多线程或者大连接的server很有可能分配的fd超过1024.

所以建议以后写的api尽量有poll或者epoll的方式。
有poll改写了下面的代码:

int waittime = (timeout->tv_sec*1000)+(timeout->tv_usec/1000.0);
struct pollfd  client;
client.fd = m_iSocket;
client.events = POLLIN;
return poll(&client, 1, waittime);

select句柄数限制 及总结相关推荐

  1. Nginx windows 版本 修改句柄数 解决 maximum number of descriptors supported by select() is 1024 while waiting

    这几天在项目上遇到个关于Nginx的问题,就是在测试中使用windows 的Nginx使用(主要用来做Socket的负载均衡),测试过程中没有发现问题(主要是测试的量太少,生产环境中差不多有1000个 ...

  2. mysql最大句柄数_【mysql】最大连接数和最大句柄数的查询和修改

    查看mysql查询最大连接数 show variables like'%connections%' :mysql查询最大连接数 set global max_connections=5000: 修改最 ...

  3. linux下查看系统进程占用的句柄数

    ---查看系统默认的最大文件句柄数,系统默认是1024 # ulimit -n 1024 ----查看当前进程打开了多少句柄数 # lsof -n|awk '{print $2}'|sort|uniq ...

  4. Linux运维:ulimit命令修改打开文件的句柄数

    文章目录 ulimit命令语法 参数说明: 查看本机设置的参数: ulimit修改打开文件的句柄数 场景 解释: 修改 1.修改file-max 2.修改ulimit的open file 注意事项 d ...

  5. python查看系统句柄数量_linux下查看系统进程占用的句柄数方法

    ---查看系统默认的最大文件句柄数,系统默认是1024 # ulimit -n 1024 ----查看当前进程打开了多少句柄数 # lsof -n|awk '{print $2}'|sort|uniq ...

  6. linux修改文件句柄数生效_linux修改句柄数

    linux服务器大并发调优时,往往需要预先调优linux参数,其中修改linux最大文件句柄数是最常修改的参数之一. 在linux中执行ulimit -a 即可查询linux相关的参数,如下所示: [ ...

  7. 服务器修改文件句柄数,请问如何修改文件最大句柄数?

    参考回答: linux默认最大文件句柄数是1024个,在linux服务器文件并发量比较大的情况下,系统会报"too many open files"的错误.故在linux服务器高并 ...

  8. Windows句柄数的限制

    最近发现以前写的部分程序存在句柄数不断增加,系统运行一段时间后就会出现问题 检查发现这是由于创建线程完成以后没有调用CloseHandle,导致句柄数量不断增加,而操作系统对句柄最大数量是有限制的,经 ...

  9. 查看进程占用的句柄数

    1.每个进程所能占用的句柄数是有上限的,如果程序打开文件.socket等,但是不关闭,并且频繁的做这样的操作,那么可能导致进程占用的句柄数超过显示,程序会异常崩溃,或产生打不开文件.socket的操作 ...

最新文章

  1. C#网络编程系列文章索引
  2. 系统架构师学习笔记_第五章(下)_连载
  3. Xilinx Altera FPGA中的逻辑资源(Slices VS LE)比较
  4. jforum oracle报错,JForum安装在Oracle数据库上
  5. 安装maven到Eclipse
  6. java页面请求跑批处理sql的有关问题
  7. Objective-C语法汇总
  8. (已解决)登录火狐浏览器账号后没有同步数据--博主的奇妙寻号之旅
  9. python测试题 --- 列表元素处理
  10. ubuntu12的程序问题
  11. thinkphp 响应
  12. 数据库笔记08:实现索引
  13. 新建计算机管理员用户,电脑我设置个新的用户 怎么切换不到默认的管理员用户登陆了...
  14. mysql 数据库授权
  15. php 即时通讯插件,四款可以整合到网站的网页版即时通信IM插件系统
  16. Fandis COSTECH A17M23SWB MT0再见,故宫
  17. 详细理解script标签
  18. 【VSCode】解决Java项目的Referenced Libraries (Read-only)问题
  19. 触龙——可解释推荐系统
  20. 12C创建用户问题 ORA-65096

热门文章

  1. 【爬虫】从零开始使用 Scrapy
  2. Docker 入门(二)常用命令纯手敲带测试结果
  3. 使用百度云盘快速下载文件
  4. 保险也能拼购?个人可以凑够人数组团购买医疗保险的4大风险
  5. WebGL入门(六)-通过坐标计算实现图形的平移、旋转和缩放变换
  6. 北航计算机考博经验,最新的北航考博经验
  7. 打印等腰三角形图案[1]
  8. 微信扫码---进入小程序指定页面(传参)
  9. 医院信息系统 固定资产管理子系统
  10. Javascript中删除数组中重复出现的元素