(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)边沿触发实例:
复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60#include <stdio.h> #include <stdlib.h> #include <sys/epoll.h> #include <errno.h> #include <unistd.h> #define MAXLINE 10 int 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 -- aaaaa buf[i-1] = 'n'; // buf -- aaaan ch++; // 'a'--'b' for (; i < MAXLINE; i++) // 从 buf[5]开始 buf[i] = ch; //buf -- aaaanbbbbb buf[i-1] = 'n'; //buf -- aaaanbbbbn ch++; // 'b'--'c write(pfd[1], buf, sizeof(buf)); // 将buf写入 管道 --- aaaanbbbbn sleep(5); } // 周期的向管道内写,再写就是 ccccnddddn close(pfd[1]); } else if (pid > 0) // 父进程读操作 (模拟一个服务器的操作) { struct epoll_event event; struct epoll_event resevent[10]; // epoll_wait就绪返回event int 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 模式)的全部内容,更多相关epoll内容请搜索靠谱客的其他文章。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复