linux select函数详解

linux select函数详解html

在Linux中,咱们可使用select函数实现I/O端口的复用,传递给 select函数的参数会告诉内核:linux

      •咱们所关心的文件描述符数组

      •对每一个描述符,咱们所关心的状态。(咱们是要想从一个文件描述符中读或者写,仍是关注一个描述符中是否出现异常)服务器

      •咱们要等待多长时间。(咱们能够等待无限长的时间,等待固定的一段时间,或者根本就不等待)数据结构

   从 select函数返回后,内核告诉咱们一下信息:函数

      •对咱们的要求已经作好准备的描述符的个数测试

      •对于三种条件哪些描述符已经作好准备.(读,写,异常)spa

   有了这些返回信息,咱们能够调用合适的I/O函数(一般是 read 或 write),而且这些函数不会再阻塞..net

 

#include <sys/select.h>   指针

    int select(int maxfdp1, fd_set *readset, fd_set *writeset, fd_set *exceptset,struct timeval *timeout);

   

   返回:作好准备的文件描述符的个数,超时为0,错误为 -1.

   

   首先咱们先看一下最后一个参数。它指明咱们要等待的时间:

struct timeval{      

        long tv_sec;   /*秒 */

        long tv_usec;  /*微秒 */   

    }

   

   有三种状况:

    timeout == NULL  等待无限长的时间。等待能够被一个信号中断。当有一个描述符作好准备或者是捕获到一个信号时函数会返回。若是捕获到一个信号, select函数将返回 -1,并将变量 erro设为EINTR。

    timeout->tv_sec == 0 &&timeout->tv_usec == 0不等待,直接返回。加入描述符集的描述符都会被测试,而且返回知足要求的描述符的个数。这种方法经过轮询,无阻塞地得到了多个文件描述符状态。

    timeout->tv_sec !=0 ||timeout->tv_usec!= 0 等待指定的时间。当有描述符符合条件或者超过超时时间的话,函数返回。在超时时间即将用完但又没有描述符合条件的话,返回 0。对于第一种状况,等待也会被信号所中断。

   

   中间的三个参数 readset, writset, exceptset,指向描述符集。这些参数指明了咱们关心哪些描述符,和须要知足什么条件(可写,可读,异常)。一个文件描述集保存在 fd_set 类型中。fd_set类型变量每一位表明了一个描述符。咱们也能够认为它只是一个由不少二进制位构成的数组。以下图所示:

   

   对于 fd_set类型的变量咱们所能作的就是声明一个变量,为变量赋一个同种类型变量的值,或者使用如下几个宏来控制它:

 

#include <sys/select.h>   

int FD_ZERO(int fd, fd_set *fdset);   

int FD_CLR(int fd, fd_set *fdset);   

int FD_SET(int fd, fd_set *fd_set);   

int FD_ISSET(int fd, fd_set *fdset);</span>

    FD_ZERO宏将一个 fd_set类型变量的全部位都设为 0,使用FD_SET将变量的某个位置位。清除某个位时可使用 FD_CLR,咱们可使用 FD_SET来测试某个位是否被置位。

   当声明了一个文件描述符集后,必须用FD_ZERO将全部位置零。以后将咱们所感兴趣的描述符所对应的位置位,操做以下:

 

fd_set rset;   

int fd;   

FD_ZERO(&rset);   

FD_SET(fd, &rset);   

FD_SET(stdin, &rset);</span>

    

    select返回后,用FD_ISSET测试给定位是否置位:

 

if(FD_ISSET(fd, &rset)   

{ ... }</span>

具体解释select的参数:

(1)intmaxfdp是一个整数值,是指集合中全部文件描述符的范围,即全部文件描述符的最大值加1,不能错。

说明:对于这个原理的解释能够看上边fd_set的详细解释,fd_set是以位图的形式来存储这些文件描述符。maxfdp也就是定义了位图中有效的位的个数。

(2)fd_set*readfds是指向fd_set结构的指针,这个集合中应该包括文件描述符,咱们是要监视这些文件描述符的读变化的,即咱们关心是否能够从这些文件中读取数据了,若是这个集合中有一个文件可读,select就会返回一个大于0的值,表示有文件可读;若是没有可读的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。能够传入NULL值,表示不关心任何文件的读变化。

(3)fd_set*writefds是指向fd_set结构的指针,这个集合中应该包括文件描述符,咱们是要监视这些文件描述符的写变化的,即咱们关心是否能够向这些文件中写入数据了,若是这个集合中有一个文件可写,select就会返回一个大于0的值,表示有文件可写,若是没有可写的文件,则根据timeout参数再判断是否超时,若超出timeout的时间,select返回0,若发生错误返回负值。能够传入NULL值,表示不关心任何文件的写变化。

(4)fd_set*errorfds同上面两个参数的意图,用来监视文件错误异常文件。

(5)structtimeval* timeout是select的超时时间,这个参数相当重要,它可使select处于三种状态,第一,若将NULL以形参传入,即不传入时间结构,就是将select置于阻塞状态,必定等到监视文件描述符集合中某个文件描述符发生变化为止;第二,若将时间值设为0秒0毫秒,就变成一个纯粹的非阻塞函数,无论文件描述符是否有变化,都马上返回继续执行,文件无变化返回0,有变化返回一个正值;第三,timeout的值大于0,这就是等待的超时时间,即 select在timeout时间内阻塞,超时时间以内有事件到来就返回了,不然在超时后无论怎样必定返回,返回值同上述。

说明:

函数返回:

(1)当监视的相应的文件描述符集中知足条件时,好比说读文件描述符集中有数据到来时,内核(I/O)根据状态修改文件描述符集,并返回一个大于0的数。

(2)当没有知足条件的文件描述符,且设置的timeval监控时间超时时,select函数会返回一个为0的值。

(3)当select返回负值时,发生错误。

理解select模型:

理解select模型的关键在于理解fd_set,为说明方便,取fd_set长度为1字节,fd_set中的每一bit能够对应一个文件描述符fd。则1字节长的fd_set最大能够对应8个fd。

(1)执行fd_set set;FD_ZERO(&set);则set用位表示是0000,0000。

(2)若fd=5,执行FD_SET(fd,&set);后set变为0001,0000(第5位置为1)

(3)若再加入fd=2,fd=1,则set变为0001,0011

(4)执行select(6,&set,0,0,0)阻塞等待

(5)若fd=1,fd=2上都发生可读事件,则select返回,此时set变为0000,0011。注意:没有事件发生的fd=5被清空。

基于上面的讨论,能够轻松得出select模型的特色:

(1)可监控的文件描述符个数取决与sizeof(fd_set)的值。我这边服务器上sizeof(fd_set)=512,每bit表示一个文件描述符,则我服务器上支持的最大文件描述符是512*8=4096。听说可调,另有说虽然可调,但调整上限受于编译内核时的变量值。

(2)将fd加入select监控集的同时,还要再使用一个数据结构array保存放到select监控集中的fd,一是用于再select返回后,array做为源数据和fd_set进行FD_ISSET判断。二是select返回后会把之前加入的但并没有事件发生的fd清空,则每次开始 select前都要从新从array取得fd逐一加入(FD_ZERO最早),扫描array的同时取得fd最大值maxfd,用于select的第一个参数。

(3)可见select模型必须在select前循环array(加fd,取maxfd),select返回后循环array(FD_ISSET判断是否有时间发生)。

基本原理

Resize icon

 select()系统调用代码走读

调用顺序以下:sys_select() à core_sys_select() à do_select() à fop->poll()

 

 

 

 

 

 

计算机生成了可选文字:stdinS七dOU七S七derrlistensocke七fdClient【]readSet[0][1]一1[2][3][FDSETSIZE一l]图1.正在监听客户端的链接

 

计算机生成了可选文字:re〕dSet115七ensocke七fdClient【][0]级IT[aXi=0S七dOU七~stdi诊‘s七derr)〔厂”rs'Cli’吐[l]一112][3][FDSETSIZE一1]其中第一个可用的套接字为4,因此第一个。liert为4,maxi为。“ent数组当前使用项的最大下标。图2.第1个客户端创建链接后

 

计算机生成了可选文字:115七ensocke七fd反firs七client眨SeCOndC"’吐厂―---readSetClient【]S七dOUt[0]…s'd'rr)一n·1d七S[l]Ir.SX立=1[3]其中第一个可用的套接字为5,因此第2个。!1ent为5,maxi为。Iient数组当前使用项的最大下标.[FDSETSIZE一1]图3.第2个客户端创建链接后.

 

计算机生成了可选文字:re日dSetClient[]“七derr〕115七ensocke七firs七clien七七ermina七ed[0]111]5i_」,_。七dou七…5'"'n)I'r己<卜=l.fdofdZfd4交SeC'ndC"’吐fds1入…d一f[2]一1一1[FDSETSIZE一1]图4.第1个客户端断开链接后

参考资料:

http://blog.csdn.net/tianmohust/article/details/6595998 

http://www.cnblogs.com/jinmu190/archive/2010/11/21/1883184.html

相关文章
相关标签/搜索