select()

select(),确定一个或多个套接口的状态,本函数用于确定一个或多个套接口的状态,对每一个套接口,调用者可查询它的可读性、可写性及错误状态信息,用fd_set结构来表示一组等待检查的套接口,在调用返回时,这个结构存有满足一定条件的套接口组的子集,并且select()返回满足条件的套接口的数目。有一组宏可用于对fd_set的操作,这些宏与Berkeley Unix软件中的兼容,但内部的表达是完全不同的。

简述

编辑
确定一个或多个 套接口的状态,如:需要则等待。
#include <sys/select.h>
int PASCAL FAR select( int nfds, fd_set FAR* readfds, fd_set FAR* writefds, fd_set FAR* exceptfds, const struct timeval FAR* timeout);
nfds:是一个整数值,是指集合中所有 文件描述符的范围,即所有文件描述符的最大值加1,不能错!在Windows中这个参数的值无所谓,可以设置不正确。
readfds:(可选) 指针,指向一组等待可读性检查的套接口。
writefds:(可选)指针,指向一组等待可写性检查的套接口。
exceptfds:(可选)指针,指向一组等待错误检查的套接口。
timeout:select()最多等待时间,对阻塞操作则为NULL。

注释

编辑
readfds参数标识等待可读性检查的套接口。如果该套接口正处于监听 listen()状态,则若有连接请求到达,该套接口便被标识为可读,这样一个 accept()调用保证可以无阻塞完成,对其他套接口而言,可读性意味着有排队数据供读取。或者对于SOCK_STREAM类型套接口来说,相对于该套接口的虚套接口已关闭,于是 recv()recvfrom()操作均能无阻塞完成,writefds参数标识等待可写性检查的套接口。如果一个套接口正在 connect()连接(非阻塞),可写性意味着连接顺利建立。如果套接口并未处于connect()调用中,可写性意味着 send()sendto()调用将无阻塞完成。〔但并未指出这个保证在多长时间内有效,特别是在多线程环境中〕。
exceptfds参数标识等待带外数据存在性或意味错误条件检查的套接口,请注意如果设置了SO_OOBINLINE选项为假FALSE,则只能用这种方法来检查带外数据的存在与否,对于SO_STREAM类型套接口,远端造成的连接中止和KEEPALIVE错误都将被作为意味出错。如果套接口正在进行连接connect()(非阻塞方式),则连接试图的失败将会表现在exceptfds参数中。
如果对readfds、writefds或exceptfds中任一个组类不感兴趣,可将它置为空NULL。
在socket.h头文件中共定义了四个宏来操作描述字集。FD_SETSIZE变量用于确定一个集合中最多有多少描述字(FD_SETSIZE缺省值为64,可在包含socket.h前用#define FD_SETSIZE来改变该值)。对于内部表示,fd_set被表示成一个套接口的队列,最后一个有效元素的后续元素为INVAL_SOCKET。宏为:FD_CLR(s,*set):从集合set中删除描述字s。 FD_ISSET(s,*set):若s为集合中一员,非零;否则为零。 FD_SET(s,*set):向集合添加描述字s。FD_ZERO(*set):将set初始化为空集NULL。
timeout参数控制select完成的时间。若timeout参数为空指针,则select将一直阻塞到有一个描述字满足条件,否则的话,timeout指向一个 timeval结构,其中指定了select调用在返回前等待多长时间。如果timeval为{0,0},则select立即返回,这可用于探询所选套接口的状态,如果处于这种状态,则select调用可认为是非阻塞的,且一切适用于非阻塞调用的假设都适用于它,举例来说,阻塞 钩子函数不应被调用,且 套接口实现不应yield。

返回值

编辑
select()调用返回处于就绪状态并且已经包含在fd_set结构中的描述字总数;如果超时则返回0;否则的话,返回SOCKET_ERROR错误,应用程序可通过WSAGetLastError获取相应错误代码。
当返回为-1时,所有描述符集清0。
当返回为0时,表示超时。
当返回为正数时,表示已经准备好的描述符数。
select()返回后,在3个描述符集里,依旧是1的位就是准备好的描述符。这也就是为什么,每次用select后都要用FD_ISSET的原因。

错误代码

编辑
WSANOTINITIALISED:在使用此 API之前应首先成功地调用WSAStartup()。
WSAENETDOWN: 套接口实现检测到网络子系统失效。
WSAEINVAL:超时时间值非法,
WSAEINTR:通过一个 WSACancelBlockingCall()来取消一个阻塞的
WSAEINPROGRESS:一个阻塞的套接口调用正在运行中。
WSAENOTSOCK:描述字集合中包含有非套接口的元素。
//下面是示例代码:
//代码是服务器TCP模型,采用多路复用的select函数实现了循环的监听并接受客户端的功能,其中也包含了上传下载的功能*/
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/socket.h>
#include<string.h>
#include<netinet/in.h>
#include<sys/ioctl.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/types.h>
#include<dirent.h>
 
int  main()
{
     struct  sockaddr_in seraddr,cliaddr;
     int  listenfd,connfd,fd1,fd2,n,m,l,port;
     char  user[20],buf[4096];
     fd_set  readfds,tmpfds; //岗哨监控集合
     socklen_t addrlen;
     DIR *dr;
     struct  dirent *file;
 
     printf ( "请输入需要设定的服务器名称:" );
     scanf ( "%s" ,user);
     
     printf ( "请输入需要设定的服务器端口:" );
     scanf ( "%d" ,&port);
     getchar ();
 
     if ((listenfd=socket(AF_INET,SOCK_STREAM,0))<0)
     {
         perror ( "创建失败" );
        exit (-1);
     }
     
     
 
     /*开始设定服务器的参数地址类型,IP,PORT*/
     memset (&seraddr,0, sizeof (seraddr)); //将服务器的初值空间清空,防止转化过程有影响
     seraddr.sin_family=AF_INET;
     seraddr.sin_port=htons(port); //将得到的本地端口转换为网络字节序
     seraddr.sin_addr.s_addr=htonl(INADDR_ANY); //将得到的ip地址字符串转换为网络字节序的ip地址数值
 
     if ((bind(listenfd,( struct  sockaddr*)&seraddr, sizeof (seraddr))<0))
     {
         perror ( "绑定失败" );
        exit (-1);
     }
     printf ( "绑定创建\n" );
     if ((connfd=listen(listenfd,50))<0)
     {
         perror ( "监听失败" );
         exit (-1);
     }
     printf ( "开始监听\n" );
     FD_ZERO(&readfds); //初始化文件集
     FD_SET(listenfd,&readfds); //将需要监视的listenfd放入readfds集中
 
     while (1) //循环监听
     {
         int  nread,n;
         tmpfds=readfds; //将监视集传递给临时的监视集中,防止后续的操作修改了监视集
         if (select(FD_SETSIZE,&tmpfds,NULL,NULL,NULL)<0) //设置监视,监视tmpfds内的子fd变化,发生变化的将会被保留在tmpfds中
           {
            perror ( "监视未成功" );
             exit (-1);
          }
         
          for (fd1=0;fd1<FD_SETSIZE;fd1++) //循环找在最大范围内的fd1
         {
            if (FD_ISSET(fd1,&tmpfds)) //查找是否fd在tmpfds里面
             {
               if (fd1==listenfd) //判定fd等于监听fd,即监听fd在监视过程中出现变化被发现
               {
                   addrlen= sizeof (cliaddr);
                  connfd=accept(listenfd,( struct  sockaddr*)&cliaddr,&addrlen); //开始接收客户
                  FD_SET(connfd,&readfds); //将connfd加入监视集,监视接入的变化
                  printf ( "接入新的连接\n" );
               }
               else
                   {
                       ioctl(fd1,FIONREAD,&nread); //测试在fd中还有nread个字符需要读取
                       if (nread==0) //如果需要读取的数据为0,则关闭检测出来的fd1,并且从监视中移除
                     {
                      close(fd1);
                      FD_CLR(fd1,&readfds);
                       printf ( "移除\n" );
                     }
                       else //如果可读数据不为0,则读出来
                     {
                         int  i;
                         char  *p=NULL,*q=NULL;
                       n=read(fd1,buf,nread);
                       buf[n]=0;
                       p=buf;
                       
                       
                           if (( strncmp (p, "-get" ,4)==0))
                           {  
                              q=p+5;
                              printf ( "客户下载文件>%s" ,q);
                              if ((fd2=open(q,O_RDONLY))<0)
                                  perror ( "打开文件错误" );
                             
                              while ((m=read(fd2,buf,4096))>0)
                              {
                                  write(connfd,buf,m);
                                  
                              }
                              bzero(buf, sizeof (buf));
                              close(fd1);
                              close(fd2);
                              FD_CLR(fd1,&readfds);
                           
                           }
                           
                           if (( strncmp (p, "-up" ,3)==0))
                           {  
                              q=p+4;
                              printf ( "客户上传文件%s\n" ,buf+4);
                              if ((fd2=open(q,O_CREAT | O_WRONLY | O_APPEND ,0666))<0)
                              {
                                 perror ( "打开文件写入失败" );
                                  
                              }
                              
                              while ((m=read(connfd,buf,128))>0)
                             
                                  printf ( "%s" ,buf);
                                  write(fd2,buf,m);
                              }
                              bzero(buf, sizeof (buf));
                              close(fd1);
                              close(fd2);
                              FD_CLR(fd1,&readfds);
                           
                           }
                           
                           if (( strncmp (p, "-ls" ,3)==0))
                           {  
                              q=p+4;
                              printf ( "客户查看文件……" );
                              if ((dr=opendir(q))==NULL)
                                  perror ( "打开目录失败" );
                              while ((file=readdir(dr))!=NULL)
                              {
                                    printf ( "%s    " ,file->d_name);
                                   write(connfd,file->d_name, sizeof (file->d_name));
 
                              }
                              close(fd1);
                              close(connfd);
                              closedir(dr);
                              FD_CLR(fd1,&readfds);
                           
                           }
                       
                       printf ( "从客户收取的信息:%s\n" ,buf);
                     }
 
 
                   }
 
             } //end if 0
             
          
         } //end for 0
 
     
     } //end while0
 
 
 
 
 
 
     exit (0);
} //end main

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值