Epoll边沿触发必须使用非阻塞的原因

文章介绍了Linux的epoll机制中的水平触发(LT)和边沿触发(ET)两种模式的区别。水平触发在缓冲区有数据时会持续触发,而边沿触发只在数据新增时触发一次。在边沿触发模式下,为避免数据丢失和服务器阻塞,必须使用非阻塞IO读取所有数据。示例代码展示了如何设置和使用这两种触发方式。
摘要由CSDN通过智能技术生成

epoll有两种触发方式

水平触发(level trigger,LT)与 边沿触发(edge trigger,ET)
水平触发与边缘触发的区别:

水平触发:只要缓冲区有数据就会一直触发

边沿触发:只有在缓冲区增加数据的那一刻才会触发

下面举一个例子说明这两者的区别

/* 使用边沿触发 */
 
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/epoll.h>
 
int main(int argc, char *argv[])
{
    int epfd, nfds;
    struct epoll_event event, events[10];
    int i;
        
    epfd = epoll_create(10);
    event.data.fd = 0; /* 监听标准输入 */
    event.events = EPOLLIN | EPOLLET; /* 读监听、边缘触发 */
    //event.events = EPOLLIN; /* 读监听、边缘触发 */
    epoll_ctl(epfd, EPOLL_CTL_ADD, 0, &event);
    
    while (1) 
    {
        nfds = epoll_wait(epfd, events, 10, -1); /* 放回就绪的描述符数量 */
        
        for (i = 0; i < nfds; i++) 
        {
            if (events[i].data.fd == 0) 
            {
                printf("hello world\n");
            }
        }
    }
    
    return 0;
}
此程序的运行效果,只有当往缓冲区写入数据时才会打印。(注意:由于没有将数据从缓冲区读取,所以此时缓冲区一直有数据)

如果将 event.events = EPOLLIN | EPOLLET(边沿触发) 改为 event.events = EPOLLIN(水平触发) ,则一旦输入数据,就会循环打印

这是因为输入数据后,并没有将其从缓冲区读取出来,此时epoll是水平触发,一直检测到缓冲区有数据,所以就一直循环打印

以上就是水平触发和边沿触发的区别

下面解释为什么使用边缘触发必须使用非阻塞
在设置边缘触发时,因为每次发消息只会触发一次(不管缓存区是否还留有数据),所以必须把数据一次性读取出来,否则会影响下一次消息

下面的代码实现的是监听文件描述符,每次固定读取5个字节

先看下面这段代码

int main(int argc, char *argv[])
{
    int epfd, nfds;
    int i;
    struct epoll_event event, events[10];
    char buf[5];
    int flag;
    
    epfd = epoll_create(10);
    event.data.fd = 0; /* 监听标准输入 */
    event.events = EPOLLIN | EPOLLET; /* 读监听、边缘触发 */
    //event.events = EPOLLIN; /* 读监听、边缘触发 */
    epoll_ctl(epfd, EPOLL_CTL_ADD, 0, &event);
 
#if 0
    flag = fcntl(0, F_GETFL);
    fcntl(0, F_SETFL, flag | O_NONBLOCK);
#endif    
 
    while (1)
    {
        int i;
        int num = 0;
        char c;
        
        nfds = epoll_wait(epfd, events, 5, -1); /* 返回就绪的文件描述符 */
        
        for (i = 0; i < nfds; ++i) 
        {
            if (events[i].data.fd == STDIN_FILENO) 
            {
                for(i = 0; i < 5; i++)
                {
                    buf[i] = getc(stdin);
                    if(buf[i] == -1)
                        break;
                }
 
                printf("hello world\n");
            }
            
        }
    }
    
    return 0;
}

使用边缘触发,必须使用while循环读尽数据,否则会造成数据丢失。

如果使用阻塞读,例如本例,每次触发读取5个字节,此时getc函数是阻塞读取,这就会引起一个问题,当缓存中的数据小于5时,就会在这里阻塞等待,导致无法处理其他IO,这是非常错误的行为,在并发的服务器中,会导致服务器阻塞,无法处理其他客户端。正确的处理方法是将读取的文件描述符设置为非阻塞,循环读取,如果没有数据了就放回错误,这样就不会让服务器阻塞。

可以添加下面这两行代码,将标准输入设置为非阻塞IO

flag = fcntl(0, F_GETFL);
fcntl(0, F_SETFL, flag | O_NONBLOCK);
这就解释了为什么边沿触发必须使用非阻塞的问题。
 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值