epoll默认的事件模型:LT(水平触发):监听的fd中有数据, 就会触发epoll_wait导致---read()管道中的数据;
0、epoll小秘书只负责监听的fd[0]有读事件产生
1、epoll_create()创建1棵监听树
2、epoll_ctl()把fd[0]加入监听树, 并监听fd[0]的读事件
3、epnum = epoll_wait()监听
4、解析epnum, 是否有读事件产生
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>
#include <sys/epoll.h>
#define MAXSIZE 10
void sys_error(const char *str)
{
perror(str);
exit(1);
}
int main(void)
{
int i, n;
int ret;
char buf[MAXSIZE], ch = 'a'; // 接收读到的数据
int fd[2];
pid_t pid;
ret = pipe(fd);
if (ret == -1)
sys_error("pipe error");
pid = fork();
if (pid == -1)
sys_error("fork error");
else if (pid == 0)
{
close(fd[0]); // 子关闭读端
// 子进程写到管道里
while (1)
{
for (i = 0; i < MAXSIZE / 2; ++i)
buf[i] = ch;
buf[i - 1] = '\n';
ch++;
for (; i < MAXSIZE; ++i)
buf[i] = ch;
buf[i - 1] = '\n';
ch++;
write(fd[1], buf, sizeof(buf));
sleep(5);
}
}
else
{
close(fd[1]); // 父关闭写端, 形成单向通路
// select poll epoll监听的是文件描述符(pipe fifo mmap, socket都可以用)
// 父进程从管道里读(read读管道会阻塞的)--引入epoll小秘书监听
// 1、创建1棵监听树
int epfd = epoll_create(128);
if (epfd == -1)
sys_error("epoll_create error");
// 2、fd[0]文件描述符加入到监听树
struct epoll_event tmp;
tmp.events = EPOLLIN; // 只监听fd[0]的读事件(默认LT水平触发)
// LT水平触发, 只要监听的文件描述符中有数据---就会触发epoll
tmp.data.fd = fd[0];
int ret = epoll_ctl(epfd, EPOLL_CTL_ADD, fd[0], &tmp);
if (ret == -1)
sys_error("epoll_ctl error");
// 3、epoll小秘书负责监听(1次调用, 监听1次)
struct epoll_event wevent[1]; // 用来接收发生了读事件(传出的)文件描述符
int epnum;
while (1)
{
epnum = epoll_wait(epfd, wevent, 1, 0); // 非阻塞轮询
if (epnum == -1)
sys_error("epoll_wait error");
else if (epnum > 0)
{
// 4、有读事件发生--read()从管道中读数据
n = read(fd[0], buf, MAXSIZE / 2);
write(STDOUT_FILENO, buf, n);
}
}
close(fd[0]);
close(epfd);
}
return 0;
}