unix环境高级编程中也有这个函数,都是为了讲解IO多路转接。从本质上来看epoll就是一个改善了的select和poll,本质没发生任何变化,对于构建在poll,select和epoll上的框架使用者来说,没什么区别。而对框架设计者来说,使用epoll来替换select或者poll代价也不大。
epoll主要涉及到三个函数,这个要比select和poll复杂,从本质上来说是因为epoll把通知内核我想要观察的描述符和内核通知我哪些描述符发生了变化分成了两个独立的部分,而前述的select和poll都是合二为一的,这也是为什么epoll能做到性能优势。
epollcreate用于创建一个文件描述符,以后所有的操作都与此文件描述符有关。实际上你可以认为这个相当于一个文件,存放了进程要观察的所有文件描述符集合。
原型是
int epoll_create(int size)
调用方式也很简单:
int epfd=epoll_create(0);
size 无需指明.
epoll_ctl的使用比较复杂,主要是涉及到epoll_event 的操作.
epoll_event的定义如下:
typedef union epoll_data {
void *ptr;
int fd;
__uint32_t u32;
__uint64_t u64;
} epoll_data_t;
//感兴趣的事件和被触发的事件
struct epoll_event {
__uint32_t events; /* Epoll events */
epoll_data_t data; /* User data variable */
};
epoll_ctl的原型如下:
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
经过以上处理,内核已经知道了和epfd关联的待观察文件描述符集以及上面的事件.
epoll_wait就是用来等待上述事件发生的,原型:int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout)
, epfd和epoll_ctl是一样的,events是内核返回给进程哪个文件描述符发生了什么事情,具体用法参考epoll_ctl。maxevents指明了events数组 大小;timeout可以指定超时,单位为毫秒(0会立即返回,-1是永久阻塞)。
这个函数同时监听标准输入和一个连接到服务器的socket,如果用户输入了什么内容就会发送到服务器,如果从服务器收到了任何内容就直接输出到终端.
//
// Created by bai on 2/2/16.
//
#include "../lib/unp.h"
#include <sys/epoll.h>
void str_cli(FILE *fp, int sockfd)
{
int efd;
struct epoll_event event;
struct epoll_event events[20];
int i;
int nfds;
char buf[MAXLINE];
int n;
efd = epoll_create (0);
if(efd<0){
err_sys("epoll create failed");
}
event.data.fd=fileno(fp);
event.events=EPOLLIN;
epoll_ctl(efd,EPOLL_CTL_ADD,fileno(fp),&event);
event.data.fd=sockfd;
event.events=EPOLLIN;
epoll_ctl(efd,EPOLL_CTL_ADD,sockfd,&event);
for ( ; ; ) {
nfds=epoll_wait(efd,events,sizeof(events)/sizeof(struct epoll_event),-1);
if(nfds<0){
err_sys("epoll wait error %d ",errno);
}
for(i=0;i<nfds;i++){
if(events[i].data.fd==sockfd){
if ( (n = Read(sockfd, buf, MAXLINE)) == 0) {
err_quit("str_cli: server terminated prematurely");
}
Write(fileno(stdout), buf, n);
}
if(events[i].data.fd==fileno(fp)){
if ( (n = Read(fileno(fp), buf, MAXLINE)) == 0) {
close(sockfd);
break;
}
Writen(sockfd, buf, n);
}
}
}
close(efd);
}