select函数在前面转载的文章中介绍过,不在赘述,下面说一下poll()函数
poll:
poll函数和select函数功能十分相似,函数声明:
int poll(struct pollfd fds[],nfds_t nfds, int timeout);
参数说明:
fds:是一个struct pollfd结构类型的数组,用于存放需要检测其状态的Socket描述符;每当调用这个函数之后,系统不会清空这个数组,操作起来比较方便;特别是于socket 连 接比较多的情况下,在一定程度上可以提高处理的效率;这一点与select()函数不同,调用select()函数之后,select()函数会清空它所检测的socket描述符集合,导致每次调用select()之前都必须把socket描述符重新加入到待检测的集合中;因此,select()函数适合于只检测一个socket描述符的情况,而poll()函数适合于大量socket描述符的情况;
nfds:nfds_t类型的参数,用于标记数组fds中的结构体元素的总数量;
timeout:是poll函数调用阻塞的时间,单位:毫秒;
返回值:
>0:数组fds中准备好读、写或出错状态的那些socket描述符的总数量;
==0:数组fds中没有任何socket描述符准备好读、写,或出错;此时poll超时,超时时间 是timeout毫秒;换句话说,如果所检测的socket描述符上没有任何事件发生的话,那么poll()函数会阻塞timeout所指定的毫秒时间长度之后返回,如果timeout==0,那么poll() 函数立即返回而不阻塞,如果timeout==INFTIM,那么poll() 函数会一直阻塞下 去,直到所检测的socket描述符上的感兴趣的事件发生是才返回,如果感兴趣的事件永远不发生,那么poll()就会永远阻塞下去;
-1: poll函数调用失败,同时会自动设置全局变量errno;
struct pollfd结构体介绍:
structpollfd {
int fd; /*文件描述符*/
short events; /* 等待的需要测试事件 */
short revents; /*实际发生了的事件,也就是返回结果 */
};
poll的events和revents标志
标志名 | events? | revents? | 说明 |
POLLIN POLLRDNORM POLLRDBAND POLLPRI | • • • • | • • • • | 不阻塞的可读除高优先级外的数据(等效于POLLRDNORM| POLLRDBAND) 不阻塞的可读普通数据(优先级波段为0) 不阻塞的可读非0优先级波段的数据 不阻塞的可读高优先级数据 |
POLLOUT POLLWRNORM POLLWRBAND | • • • | • • • | 不阻塞的可写普通数据 同上 不阻塞的可写非0优先级波段的数据 |
POLLERR POLLHUP POLLNVAL |
| • • • | 已出错 已挂断 描述符不引用一打开文件 |
应用举例:
#define POLL_FD_NUM 2
#define POLL_TIMEOUT 500 /*ms*/
unsigned char cmd_buf[512];
struct pollfd poll_fds[POLL_FD_NUM];
poll_fds[0].fd = app_fds->track_fd;
poll_fds[0].events = POLLIN | POLLERR; /*关心读取和出错事件*/
poll_fds[1].fd = app_fds->panel_fd;
poll_fds[1].events = POLLIN | POLLERR; /*关心读取和出错事件*/
while(1)
{
switch(poll((struct pollfd *)&poll_fds, POLL_FD_NUM, POLL_TIMEOUT))
{
case -1:
perror("poll error");
break;
case 0:
printf(" poll timeout\n");
break;
default:
for(i = 0; i < POLL_FD_NUM; i++)
{
if(poll_fds[i].revents & POLLIN) /*检查是否可读*/
{
read(poll_fds[i], cmd_buf, sizeof(cmd_buf));
}
else if(poll_fds[i].revents & POLLERR)
{
perror("poll device error");
}
}
break;
}
}