失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > epoll 边沿触发(ET 模式)和水平触发(LT 模式)

epoll 边沿触发(ET 模式)和水平触发(LT 模式)

时间:2021-09-20 17:25:57

相关推荐

epoll 边沿触发(ET 模式)和水平触发(LT 模式)

(1)应用 ET 模式的目的:改变 epoll_wait 的默认属性,可以减少调用 epoll_wait 函数的调用次数。

(2)思想由来:模拟电路的高低电频的思想。

水平触发: 持续的 1 持续的 0

边沿触发: 0 ->1 ;1 -> 0

(3)场景:

用 epoll 实现一个服务器,调用 epoll_wait 函数进行监听,此时 client 给 epoll 发送了 100 字节的数据,而 server 使用 read 函数读走 50 字节,剩余 50 字节,问题:epoll_wait 函数还触发么?

对于水平触发来说应该触发: 因为缓存区内还有数据没读完,你需要一直告知我。

2对于边沿触发来说不应该触发:因为 epoll_wait 的职责是告知 server 是否有读事件发生,我已经告知一次了,但是你有没有读完,你的事,我不管 。

(4)使用:默认是水平触发,若设置边沿触发,需要对监听事件的基础上加宏 EPOLLET。

(5)边沿触发实例:

#include <stdio.h>#include <stdlib.h>#include <sys/epoll.h>#include <errno.h>#include <unistd.h>#define MAXLINE 10int main(int argc, char *argv[]){int efd, i; // efd -- epoll树根,i -- 循环因子char buf[MAXLINE], ch = 'a';int pfd[2]; pipe(pfd); // 创建管道 pfd[0]--读 pfd[1]--写pid_t pid = fork();if (pid == 0) // 子进程 写操作(模拟一个客户端的操作){close(pfd[0]); // 关闭读端while (1) {for (i = 0; i < MAXLINE/2; i++)buf[i] = ch; // buf -- aaaaabuf[i-1] = '\n'; // buf -- aaaa\nch++; // 'a'--'b'for (; i < MAXLINE; i++) // 从 buf[5]开始buf[i] = ch; //buf -- aaaa\nbbbbbbuf[i-1] = '\n'; //buf -- aaaa\nbbbb\nch++; // 'b'--'cwrite(pfd[1], buf, sizeof(buf)); // 将buf写入 管道 --- aaaa\nbbbb\nsleep(5);} // 周期的向管道内写,再写就是 cccc\ndddd\nclose(pfd[1]);}else if (pid > 0) // 父进程读操作 (模拟一个服务器的操作){struct epoll_event event;struct epoll_event resevent[10]; // epoll_wait就绪返回eventint res, len;close(pfd[1]); // 关闭写端efd = epoll_create(10); // 创建epoll红黑树根event.events = EPOLLIN | EPOLLET; // ET 边沿触发 -- epoll_wait只告知server一次// 注意:event.events = EPOLLIN; // LT 水平触发 (默认)event.data.fd = pfd[0];epoll_ctl(efd, EPOLL_CTL_ADD, pfd[0], &event); // 上树while (1) {res = epoll_wait(efd, resevent, 10, -1); // 监测读端if (resevent[0].data.fd == pfd[0]) {// 如果管道里有数据需要读len = read(pfd[0], buf, MAXLINE/2); // 边沿:只读取一半 // 如果是水平触发,则全读 write(STDOUT_FILENO, buf, len);}}close(pfd[0]); // 父进程读结束,关闭读端close(efd); // 关闭树根} else {perror("fork");exit(-1);}return 0;}

结果:

水平触发结果:

父进程 epoll_wait 阻塞(epollt 认为没数据才是读事件结束,没有次数限制)。

父进程 epoll_wait 阻塞。

边沿触发结果:

父进程 epoll_wait 阻塞(epollt 认为每读一次就是读事件结束)。

父进程 epoll_wait 阻塞。

如果觉得《epoll 边沿触发(ET 模式)和水平触发(LT 模式)》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。