epoll函数
int epoll_create(int size);
调用epoll_create方法创建一个epoll的句柄。
当创建好epoll句柄后,它就会占用一个fd值。在使用完epoll后,必须调用close函数进行关闭,否则可能导致fd被耗尽。
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
epfd:epoll_create函数的返回句柄。
op:表示动作类型。有三个宏来表示: EPOLL_CTL_ADD:注册新的fd到epfd中; * EPOLL_CTL_MOD:修改已经注册的fd的监听事件; EPOLL_CTL_DEL:从epfd中删除一个fd。
Fd:需要监听的fd。
event:告诉内核需要监听什么事件
EPOLLIN:表示对应的文件描述符可读;
EPOLLOUT:表示对应的文件描述符可写;
EPOLLPRI:表示对应的文件描述符有紧急数据可读(带外数据);
EPOLLERR:表示对应的文件描述符发生错误;
EPOLLHUP:表示对应的文件描述符被挂断;
EPOLLET:将EPOLL设为边缘触发(Edge Triggered),这是相对于水平触发(Level Triggered)而言的。
int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout);
功能:等待事件的产生,类似于select()调用。
参数:
epfd:创建的epoll文件描述符
events:用来从内核得到事件的集合,
maxevents:表示每次能处理的最大事件数,告之内核这个events有多大,这个maxevents的值不能大于创建epoll_create()时的size,
timeout:是超时时间(毫秒,0会立即返回,-1阻塞)
返回值:有数据事件的个数
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <errno.h>
#include <netinet/in.h>
#include <strings.h>
#include <sys/epoll.h>
#include <string.h>
#define PORT 6050
int main(int argc, const char *argv[])
{
int fd;
if((fd = socket(AF_INET,SOCK_STREAM,0)) < 0){
perror("socket");
exit(1);
}
struct sockaddr_in sin,cin;
sin.sin_family = AF_INET;
sin.sin_port = htons(PORT);
sin.sin_addr.s_addr = htonl(INADDR_ANY);
if(bind(fd,(struct sockaddr *)&sin,sizeof(sin)) < 0){
perror("bind");
exit(1);
}
listen(fd,5);
int epfd,count,newfd,i,ret;
char buf[32];
socklen_t len = sizeof(&cin);
struct epoll_event event; //用于添加事件,其成员events用来告知内核监听事件的类型
struct epoll_event events[10]; //有数据的事件集合
memset(events,0,10*sizeof(struct epoll_event));
event.data.fd = fd;
event.events = EPOLLIN;
epfd = epoll_create(1);
epoll_ctl(epfd,EPOLL_CTL_ADD,fd,&event);
while(1)
{
count = epoll_wait(epfd,events,10,-1);
for(i=0;i<count;i++)
{
if(events[i].data.fd == fd){
if((newfd = accept(events[i].data.fd,(struct sockaddr *)&cin,&len)) < 0){
perror("accept");
exit(1);
}
printf("new user(%d) connected!\n",newfd);
event.data.fd = newfd;
event.events = EPOLLIN;
epoll_ctl(epfd,EPOLL_CTL_ADD,newfd,&event);
}
else{
bzero(buf,sizeof(buf));
do{
ret = read(events[i].data.fd,buf,sizeof(buf));
}while(ret < 0 && errno == EINTR);
if(ret < 0){
perror("read");
exit(1);
}
else if(ret == 0){
printf("client %d exited!\n",events[i].data.fd);
close(events[i].data.fd);
epoll_ctl(epfd,EPOLL_CTL_DEL,events[i].data.fd,&event);
continue;
}
printf("client %d: %s",events[i].data.fd,buf);
}
}
}
return 0;
}