日期:2014-05-16  浏览次数:20697 次

关于I/O复用 select的函数的疑问
这个是我在UNIX网络编程中对于IO复用select函数作为TCP服务器的一段代码,有点懵了。。。

C/C++ code

int main(int argc, char *argv)
{
    int                   i, maxi, maxfd, listenfd, connfd, sockfd;
    int                   nready, client[FD_SETSIZE];    
    ssize_t               n;
    fd_set                rest, allrest;
    char                  buf[512];
    socklen_t             clilen;
    struct sockaddr_in    cliaddr, servaddr;
    
    listenfd = socket(AF_INET, SOCK_STREAM, 0);
    
    bzero(&servaddr, sizeof(servaddr)); 
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(SERV_PORT);    
  
    bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
    listen(listenfd, 5);
  
    maxfd = listenfd; /* 初始化 */
    maxi = -1;        
    for (i = 0; i < FD_SETSIZE; i++)
    {
        client[i] = -1;    
    }
  
    FD_ZERO(&allrest);
    FD_SET(listenfd, &allrest); /* 告诉内核,我们需要对哪些描述字进行测试 */
  
    for (; ;)
    { 
        rest = allrest;
      
        nready = select(maxfd+1, &rest, NULL, NULL, NULL); 
      
        if (FD_ISSET(listenfd, &rest)) /* 返回后, rest结构中的内容被重写,任何未就绪的描述字被置为0 */
        {
            clilen = sizeof(cliaddr);
         connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &clilen);
              
            for (i=0; i < FD_SETSIZE; i++)
         {
             if (client[i] < 0)
             {
            client[i] = connfd;
            break;    
              }    
          }
              
        if (i == FD_SETSIZE)
        return -1;
            
          FD_SET(connfd, &allrest); 
        if (connfd > maxfd)
        maxfd = connfd;
                    
        if (i > maxi)
        maxi = i;
                    
        if (--nready <= 0)
        {
        continue;
        }
                    
      }
      
      for (i = 0; i < maxi; i++)
      {
          printf("44444444\n");
          if ((sockfd = client[i]) < 0)
          continue;
              
          if (FD_ISSET(sockfd, &rest)) 
          {
          printf("55555555555\n");
          if ((n = read(sockfd, buf, 512)) == 0)
          {
              close(sockfd);
              FD_CLR(sockfd, &allrest);    
              client[i] = -1;
                }
           else
           {
                write(sockfd, buf, n);    
           }
              
           if (--nready <= 0)
               break;
           }    
        }
    }
}




按我的理解应该是这样的:
TCP服务器开启后,首先allrest里面只有listenfd, 故select()阻塞,等待listenfd就绪。
TCP客户开启,listenfd就绪,故select()返回,此时,nready应该为1,而rest里面对应的listenfd比特位为1。
故执行if (FD_ISSET(listenfd, &rest))里面的语句,接着accept()创建一个connfd,并将此加入到client[]里面,且执行FD_SET(connfd, &allrest)将allrest字符集的connfd比特位打开。接着记录最大maxfd值为下次的select()最准备,将client[]里面的记录数的最大下标值赋给maxi。此时,--nready = 0,故跳出本次循环,接着下次的循环。于是返回最上面执行 rest = allrest,而此时的allrest里面有先前的listenfd,还有后面的connfd,故此时的rest打开了listenfd和connf位的比特位,意味着接下来调用的select()函数将阻塞于监听描述字和已建立好的客户套接口描述字上,于是在等待两者任何一个就绪。
不知道这么理解对不对?

可是问题来了,当上述条件都完成之后,我就在客户端输入并返给服务器端时,并没有反应啊,也就是说没有执行for (i = 0; i < maxi; i++)里面的语句,按道理应该是:
我客户端有了输入并发送给了服务器,那么再服务器的select()将要返回吧,并且if (FD_ISSET(sockfd, &rest)) 也成立啊?
很奇怪诶?!

------解决方案--------------------
nready = select(maxfd+1, &rest, NULL, NULL, NULL); 

这句话在for(;;)中,maxfd会被select悄悄的改变。所以,每次select的时候maxfd不一定是你原来的