poll()函数详解

113 篇文章 175 订阅

poll提供的功能与select类似,不过在处理流设备时,它能够提供额外的信息。

  #include <poll.h>

  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函数通过在revents中设置标志肌肤POLLHUP、POLLERR和POLLNVAL来反映相关条件的存在。不需要events中对于这些标志符相关的比特位进行设置。如果fd小于0, 则events字段被忽略,而revents被置为0.标准中没有说明如何处理文件结束。文件结束可以通过revents的标识符POLLHUN或返回0字节的常规读操作来传达。即使POLLIN或POLLRDNORM指出还有数据要读,POLLHUP也可能会被设置。因此,应该在错误检验之前处理正常的读操作。

poll函数的事件标志符值

常量说明
POLLIN普通或优先级带数据可读
POLLRDNORM普通数据可读
POLLRDBAND优先级带数据可读
POLLPRI高优先级数据可读
POLLOUT普通数据可写
POLLWRNORM普通数据可写
POLLWRBAND优先级带数据可写
POLLERR发生错误
POLLHUP发生挂起
POLLNVAL描述字不是一个打开的文件

 

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

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

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

 

timeout值说明
INFTIM永远等待
0立即返回,不阻塞进程
>0等待指定数目的毫秒数


例子程序: 

在/root/pro/fd1 /root/pro/fd2中分别有内容,

1234

5678

1122

3344

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <stropts.h>
#include <sys/poll.h>
#include <sys/stropts.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <poll.h>

#define BUFSIZE 1024

int main(int argc, char *argv[])
{
    char buf[BUFSIZE];
    int bytes;
    struct pollfd *pollfd;
    int i=0;
    int nummonitor=0;
    int numready;
    int errno;
    char *str;
    if(argc != 3)
    {
        fprintf(stderr,"Usage:the argc num error\n");
        exit(1);
    }


    if((pollfd = (struct pollfd*)calloc(2, sizeof(struct pollfd))) == NULL) //为struct pollfd分配空间
           exit(1);
         for(i; i<2; i++) //初始化化struct pollfd结构
    {
        str = (char*)malloc(14*sizeof(char));        
        memcpy(str,"/root/pro/",14); 
        strcat(str,argv[i+1]);//注意,需要把路劲信息放到str中,否则opne("/root/pro/argv[i]",O_RDONLY)会出错
        printf("str=%s\n",str);//原因在于,在” “之中的argv[i]是字符串,不会用变量代替argv[i].
        (pollfd+i)->fd = open(str,O_RDONLY);
        if((pollfd+i)->fd >= 0)
        fprintf(stderr, "open (pollfd+%d)->fd:%s\n", i, argv[i+1]); 
        nummonitor++;
        (pollfd+i)->events = POLLIN;
    }
    printf("nummonitor=%d\n",nummonitor);
      
    while(nummonitor > 0)
    {
        numready = poll(pollfd, 2, -1);
        if ((numready == -1) && (errno == EINTR))
            continue;        //被信号中断,继续等待
        else if (numready == -1)
            break; //poll真正错误,推出
        printf("numready=%d\n",numready);
        for (i=0;nummonitor>0 && numready>0; i++)
        {
            if((pollfd+i)->revents & POLLIN)
            {
                
                 bytes = read(pollfd[i].fd, buf, BUFSIZE);
                numready--;
                printf("pollfd[%d]->fd read buf:\n%s \n", i, buf);
                nummonitor--;
            }
        }
    }
    for(i=0; i<nummonitor; i++)
        close(pollfd[i].fd);
    free(pollfd);
    return 0;
}

输出结果:




        

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`poll_wait` 函数是 Linux 内核中用于实现轮询等待的一个重要函数,其主要作用是将当前进程挂起,等待某个特定的事件发生,例如数据从 IO 设备上到达或者设备错误等。当事件发生时,`poll_wait` 将会唤醒等待的进程,使其可以继续执行。 下面详细介绍一下 `poll_wait` 函数的使用方法。 ## 语法 ```c void poll_wait(struct file *filp, wait_queue_head_t *queue, struct wait_queue_entry *wait); ``` ## 参数 - `filp`:待等待的文件指针; - `queue`:等待队列的头指针; - `wait`:等待队列的条目指针。 ## 功能 将当前进程加入到等待队列中,等待某个特定的事件发生。 ## 注意事项 1. `poll_wait` 函数只能在进程的上下文中使用,不能在中断上下文中使用。 2. 在调用 `poll_wait` 函数之前,必须先调用 `init_waitqueue_head` 函数对等待队列进行初始化。 3. 在等待队列中,每个等待事件都必须使用唯一的等待队列条目。 ## 示例 下面是一个示例程序,演示了如何使用 `poll_wait` 函数: ```c DEFINE_WAIT(wait); init_waitqueue_head(&wait_queue_head); // 等待事件发生 poll_wait(file, &wait_queue_head, &wait); // 判断事件是否已经发生 if (event_occurred) { // 唤醒等待队列中的进程 wake_up_interruptible(&wait_queue_head); } ``` 在上面的示例程序中,首先使用 `DEFINE_WAIT` 宏定义了一个等待队列条目,然后使用 `init_waitqueue_head` 函数对等待队列进行初始化。接着,使用 `poll_wait` 函数将当前进程加入到等待队列中,等待事件发生。当事件发生时,使用 `wake_up_interruptible` 函数唤醒等待队列中的进程,使其可以继续执行。 总之,`poll_wait` 函数是 Linux 内核中实现轮询等待的一个重要函数,它可以方便地实现进程挂起和等待事件发生的功能,是 Linux 内核中非常常用的一个函数
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值