poll() 和select()使用总结

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;			
        }
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值