复用IO还有一种就是select模型,咱们下面就来简单介绍一下select用法。socket
int select(int maxfdp, fd_set *readfds, fd_set *writefds, fd_set *errorfds, struct timeval *timeout);
//fd_set是一个集合,里面存放的是文件描述符,以下是一些能够操做fd_set的宏:
fd_set set;
FD_ZERO(&set); //清空set
FD_SET(fd, &set); //将fd加入set
FD_CLR(fd, &set); //将fd从set中清除
FD_ISSET(fd, &set); //若是fd在set中则为真,一般用来检查某个文件描述符是否在描述符集合中,而后去读、写、接收链接
复制代码
struct timeval则表明超时时间,有两个成员,一个是秒数,一个是毫秒; select参数说明:函数
socket()/bind()/listen()/select()/send()/recv()/close()ui
使用select也需先将socket设置为非阻塞的 下面演示如何使用select函数:spa
struct timeval TimeOut, *pTimeOut = NULL;
fd_set readfds, writefds;
FD_ZERO (&readfds);
FD_SET (m_nSock, &readfds);
writefds = readfds;
TimeOut.tv_sec = 5;
TimeOut.tv_usec=0;
pTimeOut = &TimeOut;
while(1)
{
if ((nRetVal = select (m_nSock + 1,&readfds, &writefds, NULL, pTimeOut)) == 0)
{
//超时
return 0;
}
else if ((nRetVal < 0) && (errno == EINTR || errno == EPIPE))
continue;
}
复制代码
若是链接断开了,select会返回1,但单纯的select返回1并不能说明链接断开了,也多是有数据可读,因此此时须要再判断一下read或者recv的返回值,若是返回0,就说明链接断开了。 伪代码以下:指针
fd_set read_set;
struct timeval t_o;
FD_ZERO(&read_set);
FD_SET(lSockFd,&read_set);
t_o.tv_sec = n;/* 超时秒数*/
ret = select(lSockFd + 1,&read_set,NULL,NULL,&t_o);
if(ret == 1)
{
count = recv(lSockFd,buf,LEN,0);
if((count == 0))
{
//说明链接断开
}
}
复制代码