Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1531281
  • 博文数量: 329
  • 博客积分: 2773
  • 博客等级: 少校
  • 技术积分: 4219
  • 用 户 组: 普通用户
  • 注册时间: 2012-02-24 14:17
个人简介

淡定从容,宁静致远

文章分类

全部博文(329)

文章存档

2016年(4)

2015年(50)

2014年(68)

2013年(45)

2012年(162)

分类: LINUX

2014-09-09 22:24:02

epoll简介

在linux的网络编程中,很长的时间都在使用select来做事件触发。在linux新的内核中,有了一种替换它的机制,就是epoll。
相比于select,epoll最大的好处在于它不会随着监听fd数目的增长而降低效率。因为在内核中的select实现中,它是采用轮询来处理的,轮询的fd数目越多,自然耗时越多。并且,在linux/posix_types.h头文件有这样的声明:
#define __FD_SETSIZE    1024
表示select最多同时监听1024个fd,当然,可以通过修改头文件再重编译内核来扩大这个数目,但这似乎并不治本。

epoll的接口非常简单,一共就三个函数:


1. int epoll_create(int size);
创建一个epoll的句柄,size用来告诉内核这个监听的数目一共有多大。这个参数不同于select()中的第一个参数,给出最大监听的fd+1的值。需要注意的是,当创建好epoll句柄后,它就是会占用一个fd值,在linux下如果查看/proc/进程id/fd/,是能够看到这个fd的,所以在使用完epoll后,必须调用close()关闭,否则可能导致fd被耗尽。


2. int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
epoll的事件注册函数,它不同与select()是在监听事件时告诉内核要监听什么类型的事件,而是在这里先注册要监听的事件类型。第一个参数是epoll_create()的返回值,第二个参数表示动作,用三个宏来表示:
EPOLL_CTL_ADD:注册新的fd到epfd中;
EPOLL_CTL_MOD:修改已经注册的fd的监听事件;
EPOLL_CTL_DEL:从epfd中删除一个fd;
第三个参数是需要监听的fd,第四个参数是告诉内核需要监听什么事,struct epoll_event结构如下:
struct epoll_event {
  __uint32_t events;  /* Epoll events */
  epoll_data_t data;  /* User data variable */
};

events可以是以下几个宏的集合:
EPOLLIN :表示对应的文件描述符可以读(包括对端SOCKET正常关闭);
EPOLLOUT:表示对应的文件描述符可以写;
EPOLLPRI:表示对应的文件描述符有紧急的数据可读(这里应该表示有带外数据到来);
EPOLLERR:表示对应的文件描述符发生错误;
EPOLLHUP:表示对应的文件描述符被挂断;
EPOLLET: 将EPOLL设为边缘触发(Edge Triggered)模式,这是相对于水平触发(Level Triggered)来说的。
EPOLLONESHOT:只监听一次事件,当监听完这次事件之后,如果还需要继续监听这个socket的话,需要再次把这个socket加入到EPOLL队列里


3. int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);
等待事件的产生,类似于select()调用。参数events用来从内核得到事件的集合,maxevents告之内核这个events有多大,这个maxevents的值不能大于创建epoll_create()时的size,参数timeout是超时时间(毫秒,0会立即返回,-1将不确定,也有说法说是永久阻塞)。该函数返回需要处理的事件数目,如返回0表示已超时。

 

下面是我在redhat9上用epoll实现的简单的C/S通信,已经运行通过了。

server.c


[c-sharp] view plaincopy
  1. #include      
  2. #include      
  3. #include      
  4. #include      
  5. #include    
  6. #include   
  7. #include   
  8. #include   
  9.  
  10. #define BUFFER_SIZE 40  
  11. #define MAX_EVENTS 10  
  12.   
  13. int main(int argc, char * argv[])     
  14. {  
  15.     int server_sockfd;// 服务器端套接字     
  16.     int client_sockfd;// 客户端套接字     
  17.     int len;     
  18.     struct sockaddr_in my_addr;   // 服务器网络地址结构体     
  19.     struct sockaddr_in remote_addr; // 客户端网络地址结构体     
  20.     int sin_size;     
  21.     char buf[BUFFER_SIZE];  // 数据传送的缓冲区     
  22.     memset(&my_addr,0,sizeof(my_addr)); // 数据初始化--清零     
  23.     my_addr.sin_family=AF_INET; // 设置为IP通信     
  24.     my_addr.sin_addr.s_addr=INADDR_ANY;// 服务器IP地址--允许连接到所有本地地址上     
  25.     my_addr.sin_port=htons(8000); // 服务器端口号     
  26.     // 创建服务器端套接字--IPv4协议,面向连接通信,TCP协议  
  27.     if((server_sockfd=socket(PF_INET,SOCK_STREAM,0))<0)     
  28.     {       
  29.         perror("socket");     
  30.         return 1;     
  31.     }     
  32.     // 将套接字绑定到服务器的网络地址上  
  33.     if (bind(server_sockfd,(struct sockaddr *)&my_addr,sizeof(struct sockaddr))<0)     
  34.     {     
  35.         perror("bind");     
  36.         return 1;     
  37.     }     
  38.     // 监听连接请求--监听队列长度为5   
  39.     listen(server_sockfd,5);     
  40.     sin_size=sizeof(struct sockaddr_in);   
  41.     // 创建一个epoll句柄  
  42.     int epoll_fd;  
  43.     epoll_fd=epoll_create(MAX_EVENTS);  
  44.     if(epoll_fd==-1)  
  45.     {  
  46.         perror("epoll_create failed");  
  47.         exit(EXIT_FAILURE);  
  48.     }  
  49.     struct epoll_event ev;// epoll事件结构体  
  50.     struct epoll_event events[MAX_EVENTS];// 事件监听队列  
  51.     ev.events=EPOLLIN;  
  52.     ev.data.fd=server_sockfd;  
  53.     // 向epoll注册server_sockfd监听事件  
  54.     if(epoll_ctl(epoll_fd,EPOLL_CTL_ADD,server_sockfd,&ev)==-1)  
  55.     {  
  56.         perror("epll_ctl:server_sockfd register failed");  
  57.         exit(EXIT_FAILURE);  
  58.     }  
  59.     int nfds;// epoll监听事件发生的个数  
  60.     // 循环接受客户端请求      
  61.     while(1)  
  62.     {  
  63.         // 等待事件发生  
  64.         nfds=epoll_wait(epoll_fd,events,MAX_EVENTS,-1);  
  65.         if(nfds==-1)  
  66.         {  
  67.             perror("start epoll_wait failed");  
  68.             exit(EXIT_FAILURE);  
  69.         }  
  70.         int i;  
  71.         for(i=0;i
  72.         {  
  73.             // 客户端有新的连接请求  
  74.             if(events[i].data.fd==server_sockfd)  
  75.             {  
  76.                 // 等待客户端连接请求到达  
  77.                 if((client_sockfd=accept(server_sockfd,(struct sockaddr *)&remote_addr,&sin_size))<0)  
  78.                 {     
  79.                     perror("accept client_sockfd failed");     
  80.                     exit(EXIT_FAILURE);  
  81.                 }  
  82.                 // 向epoll注册client_sockfd监听事件  
  83.                 ev.events=EPOLLIN;  
  84.                 ev.data.fd=client_sockfd;  
  85.                 if(epoll_ctl(epoll_fd,EPOLL_CTL_ADD,client_sockfd,&ev)==-1)  
  86.                 {  
  87.                     perror("epoll_ctl:client_sockfd register failed");  
  88.                     exit(EXIT_FAILURE);  
  89.                 }  
  90.                 printf("accept client %s/n",inet_ntoa(remote_addr.sin_addr));  
  91.             }  
  92.             // 客户端有数据发送过来  
  93.             else  
  94.             {  
  95.                 len=recv(client_sockfd,buf,BUFFER_SIZE,0);  
  96.                 if(len<0)  
  97.                 {  
  98.                     perror("receive from client failed");  
  99.                     exit(EXIT_FAILURE);  
  100.                 }  
  101.                 printf("receive from client:%s",buf);  
  102.                 send(client_sockfd,"I have received your message.",30,0);  
  103.             }  
  104.         }  
  105.     }  
  106.     return 0;     
  107. }    


 

client.c


[c-sharp] view plaincopy
  1. #include     
  2. #include     
  3. #include     
  4. #include     
  5. #include     
  6. #include   
  7. #include   
  8.   
  9. #define BUFFER_SIZE 40  
  10.   
  11. int main(int argc, char *argv[])     
  12. {     
  13.     int client_sockfd;     
  14.     int len;     
  15.     struct sockaddr_in remote_addr; // 服务器端网络地址结构体     
  16.     char buf[BUFFER_SIZE];  // 数据传送的缓冲区     
  17.     memset(&remote_addr,0,sizeof(remote_addr)); // 数据初始化--清零     
  18.     remote_addr.sin_family=AF_INET; // 设置为IP通信     
  19.     remote_addr.sin_addr.s_addr=inet_addr("127.0.0.1");// 服务器IP地址     
  20.     remote_addr.sin_port=htons(8000); // 服务器端口号     
  21.     // 创建客户端套接字--IPv4协议,面向连接通信,TCP协议   
  22.     if((client_sockfd=socket(PF_INET,SOCK_STREAM,0))<0)     
  23.     {     
  24.         perror("client socket creation failed");     
  25.         exit(EXIT_FAILURE);  
  26.     }     
  27.     // 将套接字绑定到服务器的网络地址上   
  28.     if(connect(client_sockfd,(struct sockaddr *)&remote_addr,sizeof(struct sockaddr))<0)     
  29.     {     
  30.         perror("connect to server failed");     
  31.         exit(EXIT_FAILURE);  
  32.     }    
  33.     // 循环监听服务器请求      
  34.     while(1)  
  35.     {  
  36.         printf("Please input the message:");  
  37.         scanf("%s",buf);  
  38.         // exit  
  39.         if(strcmp(buf,"exit")==0)  
  40.         {  
  41.             break;  
  42.         }  
  43.         send(client_sockfd,buf,BUFFER_SIZE,0);  
  44.         // 接收服务器端信息   
  45.         len=recv(client_sockfd,buf,BUFFER_SIZE,0);  
  46.         printf("receive from server:%s/n",buf);  
  47.         if(len<0)  
  48.         {  
  49.             perror("receive from server failed");  
  50.             exit(EXIT_FAILURE);  
  51.         }  
  52.     }  
  53.     close(client_sockfd);// 关闭套接字     
  54.     return 0;  
  55. }  


 

makefile


[c-sharp] view plaincopy
  1. #This is the makefile of EpollTest  
  2.   
  3. .PHONY:all  
  4. all:server client  
  5. server:  
  6.     gcc server.c -o server  
  7. client:  
  8.     gcc client.c -o client  
  9. clean:  


阅读(3178) | 评论(0) | 转发(4) |
给主人留下些什么吧!~~