在epoll中,epoll_wait()返回就绪事件并读取客户端发来的信息,如果发来1000字节,只读取500字节,那么读端缓存中还剩下500字节,epoll_wait()会继续阻塞等待下一个读事件还是被剩下的500字节数据触发?
利用管道pipe和epoll函数做个小实验
代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<fcntl.h>
#include<sys/types.h>
#include<errno.h>
#include<sys/epoll.h>
#define MAXLINE 10
int main(int argc,char *argv[])
{
int efd,i;
int pfd[2];
pid_t pid;
char buf1[MAXLINE],buf2[MAXLINE],ch='a';
pipe(pfd);
pid=fork();
if(pid==0)
{
close(pfd[0]);//关读端
while(1) //子进程每次发送10字节数据
{
printf("****************************************\n");
// aaaa\n
for(i=0;i<MAXLINE/2;i++)
buf1[i]=ch;
buf1[i-1]='\n';
ch++;
// bbbb\n
for(;i<MAXLINE;i++)
buf1[i]=ch;
buf1[i-1]='\n';
ch++;
printf("pipe write--\n");
write(STDOUT_FILENO,buf1,sizeof(buf1));
write(pfd[1],buf1,sizeof(buf1));
sleep(5);
}
close(pfd[5]);
}
else if(pid>0)
{
struct epoll_event event;
struct epoll_event resevent[10];
int res,len;
close(pfd[1]);//关写端
efd=epoll_create(10);//创建epoll模型
//event.events =EPOLLIN | EPOLLET;//ET 边沿触发模式
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],buf2,MAXLINE/2);
printf("epoll read---\n");
write(STDOUT_FILENO,buf2,len);
}
}
}
return 0;
}
测试结果
EPOLLIN是epoll的默认模式,水平触发,从图中看出只要可读端还有数据,那么epoll_wait()不会阻塞等待,会一直把数据读完。
修改模式为边沿触发
event.events =EPOLLIN | EPOLLET;//ET 边沿触发模式
//event.events=EPOLLIN; //LT 水平触发模式(默认)
结果如下:很明显,epoll_wait()只响应子进程写入到管道的那一次