- LevelTriggered:简称LT,当FD有数据可读时,会重复通知多次,直至数据处理完成。是epoll的默认模式
- EdgeTriggered:简称ET,当FD有数据可读时,只通知一次,不管数据是否处理完成
Level是指信号只需要处于水平,就一直会触发;而edge则是指信号为上升沿或者下降沿时触发。和电路信号有关:水平触发是level trigger 一旦触发就能维持那个level.
而边缘触发edge trigger 就是一次稍纵即逝的变化.在ET模式下,缓冲区从不可读变成可读,会唤醒应用进程,缓冲区数据变少的情况,则不会再唤醒应用进程。
首先调用epoll_wait, 会将list_head链表断开,复制数据到用户空间的events中,但是还会可能残留一些数据没有读取完。这时候,如果是ET模式,链表结构将不恢复;如果是LT模式,内核将把链表结构恢复,这样下次再调用epoll_wait的时候能够继续读。
似乎,ET模式好像没什么用,因为有残留数据的问题读不到,但是仍然有两种方式可以解决:
- 既然内核不帮我们恢复链表结构,我们自己恢复,在第一次读完数据之后,如果还有残留数据,调用epoll_ctl函数,手动添加回去
- 在第一次从fd读取数据的时候,既然每一个fd事件就绪时只通知一次,那么可以while循环一直读取数据。但是注意不能用阻塞IO,因为阻塞IO在读取完数据后还在读,会陷入死循环。因此要用非阻塞IO,如果没数据可读要返回一个标识,跳出循环。
总结:
- ET模式避免了LT模式可能出现的惊群现象
- ET模式最好结合非阻塞IO读取FD数据,相比LT会复杂一点,但是比LT模式性能好一点
惊群现象:如果有n个不同的进程同时监听某个fd,并且都在调用epoll_wait获取就绪的fd,结果fd就绪后会通知这些进程。LT模式中任何一个进程通知完,fd还会存在链表当中,因此所有监听fd的进程都会被唤醒。但是真正在处理的时候,前面一两个进程可以将fd中的数据读取完,后续被唤醒的进程就没必要。然而ET模式不会出现惊群现象。因此ET模式比LT模式好一点。