20190507(select()函数)

1.select()学习

学习select()是因为要用到了,要仿照写一个进程,

学习网站:https://blog.csdn.net/cstarbl/article/details/7645298

select函数用于在非阻塞中,当一个套接字或一组套接字有信号时通知你,系统提供select函数来实现多路复用输入/输出模型。

        #include <sys/time.h> 
        #include <unistd.h> 
        int select(int maxfd,fd_set *rdset,fd_set *wrset,fd_set *exset,struct timeval *timeout);

功能:把当前符合要求的fd(可读可写有异常)分别放入rdset,wrset,exset中;

参数:maxfd:需要监视的最大的文件描述符值+1;

          rdset:  用来检查可读性的一组文件描述字

          wrset:用来检查可写性的一组文件描述字

          exset:用来检查是否有异常条件出现的文件描述字。(注:错误不包括在异常条件之内)

          struct timeval结构用于描述一段时间长度,如果在这个时间内,需要监视的描述符没有事件发生则函数返回,返回值为0。有三种可能:

                        1.timeout=NULL(阻塞:select将一直被阻塞,直到某个文件描述符上发生了事件)
                        2.timeout所指向的结构设为非零时间(等待固定时间:如果在指定的时间段里有事件发生或者时间耗尽,函数均返回)
                        3.timeout所指向的结构,时间设为0(非阻塞:仅检测描述符集合的状态,然后立即返回,并不等待外部事件的发生)

PS:  fd_set(数据类型)是一组文件描述字(fd)的集合,它用一位来表示一个fd,

对于fd_set类型通过下面四个宏来操作:       

         FD_ZERO(fd_set *fdset);将指定的文件描述符集清空,在对文件描述符集合进行设置前,必须对其进行初始化,如果不清空,由于在系统分配内存空间后,通常并不作清空处理,所以结果是不可知的。
         FD_SET(int fd,fd_set *fdset);用于在文件描述符集合中增加一个新的文件描述符。 
         FD_CLR(int fd,fd_set *fdset);用于在文件描述符集合中删除一个文件描述符。 
         FD_ISSET(int fd,fd_set *fdset);用于测试指定的文件描述符是否在该集合中。 

         过去,一个fd_set通常只能包含小于32的fd(文件描述字),因为fd_set其实只用了一个32位矢量来表示fd;现在,UNIX系统通常会在头文件<sys/select.h>中定义常量FD_SETSIZE,它是数据类型fd_set的描述字数量,其值通常是1024,这样就能表示小于1024的fd

PS:   注意fd的最大值必须<FD_SETSIZE。

socket   s;   
.....   
fd_set   set;   
while(1)   
{       
      FD_ZERO(&set);//将你的套节字集合清空   
      FD_SET(s,   &set);//加入你感兴趣的套节字到集合,这里是一个读数据的套节字s   
      select(s+1,&set,NULL,NULL,NULL);//检查套节字是否可读,   
                                                        //很多情况下就是是否有数据(注意,只是说很多情况)  
                                                        //这里select是否出错没有写   
      if(FD_ISSET(s,   &set)   //检查s是否在这个集合里面,   
      {                                           //select将更新这个集合,把其中不可读的套节字去掉   
                                                  //只保留符合条件的套节字在这个集合里面                         
              recv(s,...);   
      }   
      //do   something   here   
}

  理解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函数的过程一般是:
    先调用宏FD_ZERO将指定的fd_set清零;

    然后调用宏FD_SET将需要测试的fd加入fd_set;

    接着调用函数select测试fd_set中的所有fd;

    最后用宏FD_ISSET检查某个fd在函数select调用后,相应位是否仍然为1。

     int isready(int fd)
     {
         int rc;
         fd_set fds;
         struct tim tv;    
         FD_ZERO(&fds);
         FD_SET(fd,&fds);
         tv.tv_sec = tv.tv_usec = 0;    
         rc = select(fd+1, &fds, NULL, NULL, &tv);
         if (rc < 0)   //error
         return -1;    
         return FD_ISSET(fd,&fds) ? 1 : 0;
     }