免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 3757 | 回复: 2
打印 上一主题 下一主题

select的FD_SET问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-01-26 18:17 |只看该作者 |倒序浏览
  fd_set mrfd,rfd;
  FD_ZERO(&mrfd);
  FD_SET(sock,&mrfd);

  step.tv_sec  = 1000;
  step.tv_usec = 500;
  maxp = sock;
  memcpy(&rfd,&mrfd,sizeof(mrfd)); //rfd is a working fd_set
  while( 1 ){
  
    ret = select(maxp+1 ,&rfd,NULL ,NULL,&step);//这里使用rfd
    if(ret < 0){
      perror("SELECT:");
      return -1;        
    }           
    if(ret == 0){
      continue;         
    }           
    ready = ret;
    for(i=3;i<=maxp && ready > 0;i++){
      if(FD_ISSET(i,&rfd) == 0)
        continue;               
      if(sock == i){   
        ready--;               
        do{                     
          newsock = accept(sock,NULL,NULL);
          if(newsock < 0){              
            if(errno != EWOULDBLOCK){           
              perror("ACCEPT:");                        
              return -1;                                
            }                                   
            continue;                           
          }                             
          else{                        
            printf("accept new conn ... %d \n",newsock);
            fcntl(newsock,F_SETFL,O_NONBLOCK);  
            FD_SET(newsock,&rfd);     //将fd加入到rfd         
            maxp = MAX(maxp,newsock);           
            continue;                           
          }                             
        }while(newsock != -1);  
      }                 
    }           
  }     


代码执行后,通过strace跟踪。当第一个连接进入后,select的rfd是3 4,当第二个连接进入后,继续fd_set rfd,但select中的rfd就变成3 5了(期待值是 3 4 5)
17:22:04 select(5, [3 4], NULL, NULL, {995, 802000}) = 1 (in [3], left {994, 189000}) <1.611893>
17:22:06 accept(3, 0, NULL)             = 5 <0.000007>
17:22:06 write(1, ""..., 23accept new conn ... 5
)            = 23 <0.000006>
17:22:06 fcntl(5, F_SETFL, O_RDONLY|O_NONBLOCK) = 0 <0.000003>
17:22:06 accept(3, 0, NULL)             = -1 EAGAIN (Resource temporarily unavailable) <0.000004>
17:22:06 select(6, [3 5], NULL, NULL, {994, 189000} <unfinished ...>17:22:04 select(5, [3 4], NULL, NULL, {995, 802000}) = 1 (in [3], left {994, 189000}) <1.611893>
17:22:06 accept(3, 0, NULL)             = 5 <0.000007>
17:22:06 write(1, ""..., 23accept new conn ... 5
)            = 23 <0.000006>
17:22:06 fcntl(5, F_SETFL, O_RDONLY|O_NONBLOCK) = 0 <0.000003>
17:22:06 accept(3, 0, NULL)             = -1 EAGAIN (Resource temporarily unavailable) <0.000004>
17:22:06 select(6, [3 5], NULL, NULL, {994, 189000} <unfinished ...>

[ 本帖最后由 朱厚照 于 2010-1-26 18:23 编辑 ]

论坛徽章:
0
2 [报告]
发表于 2010-01-26 18:22 |只看该作者
当使用二个rfd时,比较正常
  FD_ZERO(&mrfd);
fd_set mrfd,rfd;  
FD_SET(sock,&mrfd);

  step.tv_sec  = 1000;
  step.tv_usec = 500;
  maxp = sock;

  while( 1 ){
    memcpy(&rfd,&mrfd,sizeof(mrfd)); //每次将mrfd的值over rfd的值
    ret = select(maxp+1 ,&rfd,NULL ,NULL,&step); //select使用rfd
    if(ret < 0){
      perror("SELECT:");
      return -1;        
    }           
    if(ret == 0){
      continue;         
    }           
    ready = ret;
    for(i=3;i<=maxp && ready > 0;i++){
      if(FD_ISSET(i,&rfd) == 0)
        continue;               
      if(sock == i){   
        ready--;               
        do{                     
          newsock = accept(sock,NULL,NULL);
          if(newsock < 0){              
            if(errno != EWOULDBLOCK){           
              perror("ACCEPT:");                        
              return -1;                                
            }                                   
            continue;                           
          }                             
          else{                        
            printf("accept new conn ... %d \n",newsock);
            fcntl(newsock,F_SETFL,O_NONBLOCK);  
            FD_SET(newsock,&mrfd);   //这里将fd加入到mrfd,而不是rfd            
            maxp = MAX(maxp,newsock);           
            continue;                           
          }                             
        }while(newsock != -1);  
      }                 
    }           


代码改正后,select正常
17:23:56 select(4, [3], NULL, NULL, {1000, 500}) = 1 (in [3], left {995, 404000}) <4.595967>
17:24:00 accept(3, 0, NULL)             = 4 <0.000007>
17:24:00 fstat(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(136, 1), ...}) = 0 <0.000003>
17:24:00 mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x2a95557000 <0.000004>
17:24:00 write(1, ""..., 23accept new conn ... 4
)            = 23 <0.000005>
17:24:00 fcntl(4, F_SETFL, O_RDONLY|O_NONBLOCK) = 0 <0.000003>
17:24:00 accept(3, 0, NULL)             = -1 EAGAIN (Resource temporarily unavailable) <0.000004>
17:24:00 select(5, [3 4], NULL, NULL, {995, 404000}) = 1 (in [3], left {992, 463000}) <2.940178>
17:24:03 accept(3, 0, NULL)             = 5 <0.000007>
17:24:03 write(1, ""..., 23accept new conn ... 5
)            = 23 <0.000004>
17:24:03 fcntl(5, F_SETFL, O_RDONLY|O_NONBLOCK) = 0 <0.000003>
17:24:03 accept(3, 0, NULL)             = -1 EAGAIN (Resource temporarily unavailable) <0.000004>
17:24:03 select(6, [3 4 5], NULL, NULL, {992, 463000} <unfinished ...>

请教一下这是为什么呢?为什么只使用一个rfd会FD_SET错误呢

论坛徽章:
0
3 [报告]
发表于 2010-01-27 16:02 |只看该作者
ding yi xia.
up
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP