对Linux 多路复用Epoll模型的水平出发模式和边缘触发模式的理解

水平触发通知又叫“低速模式”,是linux epoll模型的默认方式。对于此方式《Linux/Unix系统编程手册(下册)》中的解释是:如果文件描述符上可以非阻塞地执行I/O系统调用,此时认为它已经就绪。

对此,我的理解是:某个I/O实践就绪,例如,文件缓冲区收到了5个字节的数据,此时此文件描述符可读,触发epoll可读事件。但是如果就绪的文件描述符是非阻塞的(也就是说中途可以被打断),当读取三个字节的时候被别的事件打断了,下次仍然触发此描述符的可读事件,会接着读上次没读完的两个字节。

边沿触发通知又叫”高速模式“(ET),书上的解释是:如果文件描述符自上次状态检查以来有了新的I/O活动(比如新的输入),此时需要触发通知。

我的理解是:拿可读事件来说,如果文件缓冲区收到了5个字节的数据,触发通知,开始读取缓冲区数据,如果此文件描述符仍然是非阻塞的,被别的信号打断后,未来得及读取的数据将丢失,只有下次再有新的数据加入缓冲区后,再次触发通知,读取新的数据。因此使用"ET模式"时需要尽可能多的读取或者写入数据。

设置EPOLL 为ET 模式的方法:

 struct epoll_event ev;
 ev.data.fd = fd;
 ev.events = EPOLLIN | EPOLLET;
 epoll_ctl(epoll_fd,EPOLL_ZTL_ADD,fd, &ev);
设置文件为非阻塞模式的方法:

主要使用如下两个函数获取文件的flags,即open函数的第二个参数:
    flags = fcntl(fd,F_GETFL,0);
设置文件的flags:
    fcntl(fd,F_SETFL,flags);

设置方法
      flags = fcntl(fd,F_GETFL,0);   
      flags |= O_NONBLOCK;     
      fcntl(fd,F_SETFL,flags);

      或者一步到位的方式:
    fcntl(socket,F_SETFL,fcntl(socket,F_GETFL)|O_NONBLOCK);

相关推荐