利用select函数可以完成非阻塞通信,它可以在用户指定的时间内监听一些句柄(文件描述符?)(一个句柄就是你给一个文件,设备,套接字(socket)或管道的一个名字, 以便帮助你记住你正处理的名字, 并隐藏某些缓存等的复杂性。),当readfds或writefds中映象的文件可读或可写或超时,本次select()就结束返回。程序员利用一组系统提供的宏在select()结束时便可判断哪一文件可读或可写。
。
函数原型:
int select(int nfds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout);
ndfs:select监视的文件句柄数,视进程中打开的文件数而定,一般设为你要监视各文件
中的最大文件号加一。
readfds:select监视的可读文件句柄集合。
writefds: select监视的可写文件句柄集合。
exceptfds:select监视的异常文件句柄集合。
timeout:本次select()的超时结束时间。
函数返回值:
0表示超时
-1表示出错
正数表示文件可读或可写
相关的宏:
FD_ZERO(fd_set *fdset):清空fdset与所有文件句柄的联系。
FD_SET(int fd, fd_set *fdset):建立文件句柄fd与fdset的联系。
FD_CLR(int fd, fd_set *fdset):清除文件句柄fd与fdset的联系。
FD_ISSET(int fd, fd_set *fdset):检查fdset联系的文件句柄fd是否可读写,当>0表示可读写。
select只能监听文件描述符(file descriptors),文件指针是不行的。通过int fileno(FILE* fp)函数可以完成文件指针到文件描述符的转换。
#include#include#include#include#include#includeintmain(){intfno=fileno(stdin);//得到sdtin的文件描述符fd_set fdR;structtimeval timeout={.tv_sec=10, .tv_usec=0};
FD_ZERO(&fdR);
FD_SET(fno,&fdR);charstr[64];intgoOn=1;while(goOn){
printf("goOning
\n");switch(select(fno+1,&fdR, NULL, NULL,&timeout)){case-1:
printf("select fail\n");
goOn=0;break;case0:
printf("select end\n");
goOn=0;break;default:
scanf("%s", str);
printf("str=%s\n", str);
}
}
}
代码片段1:在指定时间内读取stdin的数据。
代码片段2:服务器在一定时间内接收子节点发来的数据报。
intsocketfd=socket(AF_INET, SOCK_DGRAM,0);
…
…
fd_set fdR;structtimeval timeout={tv_sec:10, tv_usec:0};//等待接收一定时间之后结束intgoOn=1;
FD_ZERO(&fdR);
FD_SET(sockfd,&fdR);while(goOn){switch(select(sockfd+1,&fdR, NULL, NULL,&timeout)){case-1:
printf("select fail\n");
goOn=0;break;case0:
printf("select timeout\n");
goOn=0;break;default://正常情况,读取信息if((numbytes=recvfrom(sockfd,&cdt,sizeof(cdt),0,
(structsockaddr*)&their_addr,&addr_len))==-1){
printf("recvfrom fail\n");
}
}