poll函数

来源:互联网 发布:soundtrack pro mac 编辑:程序博客网 时间:2024/05/18 16:57

函数原型:
  int poll(struct pollfd fd[], nfds_t nfds, int timeout);
  
  参数:
  
   1)第一个参数:一个结构数组,struct pollfd结构如下:
  struct pollfd{
  int fd; //文件描述符
  short events; //请求的事件
  short revents; //实际发生的的事件
  };
  
  events和revents是通过对代表各种事件的标志进行逻辑或运算构建而成的。events包括要监视的事件,poll用已经发生的事件填充revents。

常用的poll函数的事件标志符值:

常量 及 说明
POLLIN - 普通数据可读
POLLPRI - 高优先级数据可读
POLLOUT - 普通数据可写
POLLERR - 发生错误
POLLHUP - 发生挂起

注意:后两个只能作为描述字的返回结果存储在revents中,而不能作为测试条件用于events中。

2)第二个参数nfds:要监视的描述符的数目。

3)最后一个参数timeout:是一个用毫秒表示的时间,是指定poll在返回前没有接收事件时应该等待的时间。如果 它的值为-1,poll就永远都不会超时。如果整数值为32个比特,那么最大的超时周期大约是30分钟。

timeout的值和说明:

  • 列表内容INFTIM 永远等待
  • 0 立即返回,不阻塞进程
  • >0 等待指定数目的毫秒数

返回值

大于0:表示数组fds中有socket描述符的状态发生变化,或可以读取、或可以写入、或出错。并且返回的值表示这些状态有变化的socket描述符的总数量;此时可以对fds数组进行遍历,以寻找那些revents不空的socket描述符,然后判断这个里面有哪些事件以读取数据。
等于0:表示没有socket描述符有状态变化,并且调用超时。
小于0:此时表示有错误发生,此时全局变量errno保存错误码。

#include<stdio.h>#include<poll.h>#include<string.h>//poll()函数可以用于执行多路复用I/Oint main(){    struct pollfd evs;    evs.fd=0;//poll 的文件描述符    evs.events=POLLIN;//针对文件描述符所等待的事件 位掩码 POLLIN可以在设备文件的可读情况下结束 poll() 函数    evs.revents=0;//实际发生的事件 位掩码    int timeout=3000;    while(1)    {        switch(poll(&evs,1,timeout))        {            case -1:                perror("poll");                break;            case 0:                printf("timeout...\n");                break;            defalut:                {                    char buf[1024];                    memset(buf,'\0',sizeof(buf));                    fgets(buf,sizeof(buf)-1,stdin);                    printf("msg is:%s\n",buf);                }                break;        }    }    return 0;}

总结pool()函数的优缺点:
  poll与select不同,通过一个pollfd数组向内核传递需要关注的事件,故没有描述符个数的限制, 
  pollfd中的events字段和revents分别用于标示关注的事件和发生的事件,故pollfd数组只需要被初始化一次。
  poll的实现机制与select类似,其对应内核中的sys_poll,只不过poll向内核传递pollfd数组,然后对pollfd中的每个描述符进行poll,相比处理fdset来说,poll效率更高。 
  poll返回后,需要对pollfd中的每个元素检查其revents值,来得指事件是否发生。

poll优点

1)poll() 不要求开发者计算最大文件描述符加一的大小。
2)poll() 在应付大数目的文件描述符的时候相比于select速度更快
3)它没有最大连接数的限制,原因是它是基于链表来存储的。

poll缺点

1)大量的fd的数组被整体复制于用户态和内核地址空间之间,而不管这样的复制是不是有意义。
2)与select一样,poll返回后,需要轮询pollfd来获取就绪的描述符

原创粉丝点击