epoll

简介

epoll是为处理大批量句柄而作了改进的poll,它是在2.5.44内核中被引进的。

相关函数调用

int epoll_create(int size);

创建一个epoll的句柄。自从linux2.6.8之后,size参数是被忽略的。需要注意的是,当创建好epoll句柄后,它就是会占用一个fd值,在linux下如果查看/proc/进程id/fd/,是能够看到这个fd的,所以在使用完epoll后,必须调用close()关闭,否则可能导致fd被耗尽。

注:现在更推荐使用 epoll_create1(0) 来代替普通的用法。另外epoll_create1(EPOLLCLOEXEC)表示生成的epoll fd具有“执行后关闭”特性。

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结构如下:

  1. //保存触发事件的某个文件描述符相关的数据(与具体使用方式有关)
  2. typedef union epoll_data {
  3. void *ptr;
  4. int fd;
  5. __uint32_t u32;
  6. __uint64_t u64;
  7. } epoll_data_t;
  8. //感兴趣的事件和被触发的事件
  9. struct epoll_event {
  10. __uint32_t events; /* Epoll events */
  11. epoll_data_t data; /* User data variable */
  12. };

events可以是以下几个宏的集合:

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

注:这个函数在指定EPOLL_CTL_DEL时,为了与linux内核2.6.9之前相兼容,还是要让最后的参数指向一个非null变量。

另外,events.EPOLLONESHOT确实表示只监听一次事件,但是当我们监听完这次事件之后,如果还需要继续监听这个fd的话,只需要使用EPOLL_CTL_MOD修改event。

int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);

收集在epoll监控的事件中已经发送的事件。参数events是分配好的epoll_event结构体数组,epoll将会把发生的事件赋值到events数组中(events不可以是空指针,内核只负责把数据复制到这个events数组中,不会去帮助我们在用户态中分配内存)。maxevents告之内核这个events有多大,这个 maxevents的值不能大于创建epoll_create()时的size,参数timeout是超时时间(毫秒,0会立即返回,-1将不确定,也有说法说是永久阻塞)。如果函数调用成功,返回对应I/O上已准备好的文件描述符数目,如返回0表示已超时。

工作原理

两种工作方式

  • 水平触发LT
  • 边缘触发ET

两者的区别可以用下面这个例子来理解:

如果某fd上有2kb的数据,应用程序只读了1kb,ET就不会在下一次epoll_wait的时候返回,读完以后又有新数据才返回。而LT每次都会返回这个fd,只要这个fd有数据可读。

注:ET模式必须使用非阻塞套接字,可使用如下代码设置。

  1. void setnonblocking(int sockFd)
  2. {
  3. int opt;
  4. //获取sock原来的flag
  5. opt = fcntl(sockFd, F_GETFL);
  6. if (opt < 0)
  7. {
  8. printf("fcntl(F_GETFL) fail.");
  9. exit(-1);
  10. }
  11. //设置新的flag,非阻塞
  12. opt |= O_NONBLOCK;
  13. if (fcntl(sockFd, F_SETFL, opt) < 0)
  14. {
  15. printf("fcntl(F_SETFL) fail.");
  16. exit(-1);
  17. }
  18. }

示例

使用epoll实现的echo服务器。

  1. #include <unp.h>
  2. #include <sys/epoll.h>
  3. #define EPOLL_SIZE 20 //epoll关注的最大fd数目
  4. #define EVENT_ARR 10 //事件数
  5. void setnonblocking(int sockFd)
  6. {
  7. int opt;
  8. //获取sock原来的flag
  9. opt = fcntl(sockFd, F_GETFL);
  10. if (opt < 0)
  11. {
  12. printf("fcntl(F_GETFL) fail.");
  13. exit(-1);
  14. }
  15. //设置新的flag,非阻塞
  16. opt |= O_NONBLOCK;
  17. if (fcntl(sockFd, F_SETFL, opt) < 0)
  18. {
  19. printf("fcntl(F_SETFL) fail.");
  20. exit(-1);
  21. }
  22. }
  23. int main(int argc,char** argv)
  24. {
  25. int listenfd=openListenfd(SERV_PORT);
  26. setnonblocking(listenfd); //设置socket为非阻塞
  27. //创建epoll
  28. int epfd=epoll_create(EPOLL_SIZE);
  29. struct epoll_event ev,evs[EVENT_ARR];
  30. //绑定listenfd
  31. ev.data.fd=listenfd;
  32. ev.events=EPOLLIN|EPOLLET; //ET模式下必须设置为非阻塞
  33. epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&ev);
  34. int clientfd;
  35. struct sockaddr_in clientaddr;
  36. socklen_t clientlen;
  37. char buf[1024];
  38. while(1)
  39. {
  40. int nfds=epoll_wait(epfd,evs,EVENT_ARR,-1);
  41. for (int i=0;i<nfds;++i) //遍历取到的事件
  42. {
  43. //如果为listenfd的事件,且为可读
  44. if (evs[i].data.fd==listenfd && evs[i].events & EPOLLIN)
  45. {
  46. clientlen=sizeof(clientaddr);
  47. clientfd=Accept(listenfd,(SA*)&clientaddr,&clientlen);
  48. setnonblocking(clientfd);
  49. //注册客户端fd事件
  50. ev.data.fd=clientfd;
  51. ev.events=EPOLLIN|EPOLLET;
  52. epoll_ctl(epfd,EPOLL_CTL_ADD,clientfd,&ev);
  53. }
  54. else if (evs[i].events & EPOLLIN)
  55. {
  56. if ((clientfd=evs[i].data.fd)>0)
  57. {
  58. int len=read(clientfd,buf,BUF_SIZE);
  59. if (len>0)
  60. {
  61. do
  62. {
  63. if (write(clientfd, buf, len) < 0)
  64. perror("write() fail.\n");
  65. len = read(clientfd, buf, BUF_SIZE);
  66. } while (len > 0);
  67. }
  68. else if (len == 0) //出现EPOLLIN但无数据,说明断线
  69. {
  70. printf("client closed at %d\n", clientfd);
  71. epoll_ctl(epfd, EPOLL_CTL_DEL, clientfd, &ev);
  72. close(clientfd);
  73. evs[i].data.fd = -1;
  74. break;
  75. }
  76. else if (len == EAGAIN)
  77. {
  78. continue;
  79. }
  80. }
  81. }
  82. else
  83. {
  84. printf("other event\n");
  85. }
  86. }
  87. }
  88. return 0;
  89. }

参考自:

http://blog.csdn.net/xiajun07061225/article/details/9250579

http://www.cnblogs.com/aicro/archive/2012/12/27/2836170.html

http://lifeofzjs.com/blog/2015/05/16/how-to-write-a-server/

I/O复用之epoll的更多相关文章

  1. Linux下的I/O复用与epoll详解(转载)

    Linux下的I/O复用与epoll详解 转载自:https://www.cnblogs.com/lojunren/p/3856290.html  前言 I/O多路复用有很多种实现.在linux上,2 ...

  2. 从I/O复用谈epoll为什么高效

    上一篇文章中,谈了一些网络编程的基本概念.在现实使用中,用的最多的就是I/O复用了,无非就是select,poll,epoll 很多人提到网络就说epoll,认为epoll效率是最高的.单纯的这么认为 ...

  3. Linux下的I/O复用与epoll详解

    前言 I/O多路复用有很多种实现.在linux上,2.4内核前主要是select和poll,自Linux 2.6内核正式引入epoll以来,epoll已经成为了目前实现高性能网络服务器的必备技术.尽管 ...

  4. 并发服务器--02(基于I/O复用——运用epoll技术)

    本文承接自上一博文I/O复用——运用Select函数. epoll介绍 epoll是在2.6内核中提出的.和select类似,它也是一种I/O复用技术,是之前的select和poll的增强版本. Li ...

  5. I/O复用及epoll基础知识

    IO multiplexing IO multiplexing这个词可能有点陌生,但是如果我说select,epoll,大概就都能明白了.有些地方也称这种IO方式为event driven IO.我们 ...

  6. 深入理解Linux的I/O复用之epoll机制

    0.概述 通过本篇文章将了解到以下内容: I/O复用的定义和产生背景 Linux系统的I/O复用工具演进 epoll设计的基本构成 epoll高性能的底层实现 epoll的ET模式和LT模式 epol ...

  7. 多线程 or I/O复用select/epoll

    1:多线程模型适用于处理短连接,且连接的打开关闭非常频繁的情形,但不适合处理长连接.线程模型默认情况下,在Linux下每个线程会开8M的栈空间,在TCP长连接的情况下,以2000/分钟的请求为例,几乎 ...

  8. IO复用之epoll系列

    epoll是什么? epoll是Linux内核为处理大批量文件描述符而作了改进的poll,是Linux下多路复用IO接口select/poll的增强版本,它能显著提高程序在大量并发连接中只有少量活跃的 ...

  9. Linux I/O复用中select poll epoll模型的介绍及其优缺点的比較

    关于I/O多路复用: I/O多路复用(又被称为"事件驱动"),首先要理解的是.操作系统为你提供了一个功能.当你的某个socket可读或者可写的时候.它能够给你一个通知.这样当配合非 ...

随机推荐

  1. python中enumerate、变量类型转换

    enumerate可以在遍历过程中自动生成新的一列并从0开始计数 1 a = ["hello", "world", "dlrb"] 2 fo ...

  2. Android自定义View学习(四)

    硬件加速 参考:HenCoder Android 自定义 View 1-8 硬件加速 硬件加速能够让绘制变快,主要有三个原因: 本来由 CPU 自己来做的事,分摊给了 GPU 一部分,自然可以提高效率 ...

  3. NIO,OIO,AIO区别

    OIO中,每个线程只能处理一个channel(同步的,该线程和该channel绑定). 线程发起IO请求,不管内核是否准备好IO操作,从发起请求起,线程一直阻塞,直到操作完成,如图: NIO中,每个线 ...

  4. Flex的一些小实例

    1,以上是一个导航菜单 2一下是一个撑开的mx:Spacer

  5. mezzanine的page_menu tag

    mezzanine的head 导航条.左侧tree.footer是由page_menu产生的.page_menu的算法,先计算出每一页的孩子,然后再逐页去page_menu. @register.re ...

  6. Mysql 视图使用

    视图 简单理解视图就是一张虚拟表,可以简化一些复杂查询语句 举个简单的例子来理解视图 视图是虚拟的表,与包含数据的表不一样,视图只包含使用时动态检索数据的查询:不包含任何列或数据.使用视图可以简化复杂 ...

  7. Vue.js——使用$.ajax和vue-resource实现OAuth的注册、登录、注销和API调用

    转自:https://www.cnblogs.com/keepfool/p/5665953.html 概述 上一篇我们介绍了如何使用vue resource处理HTTP请求,结合服务端的REST AP ...

  8. 判断窗体 show完成

    TForm窗体 this->Visible           this->Showing        true: 显示 fale:关闭     这个可用    this->Act ...

  9. day21-类的组合

    一.面向对象的组合用法 软件重用的重要方式除了继承之外还有另外一种方式,即:组合组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合 用组合的方式建立了类与组合的类之间的关系,它是一种‘ ...

  10. LeetCode 题解 56. Merge Intervals

    题目大意:给出一组区间,合并他们. 首先是排序,首先看start,start小的在前面.start相同的话,end小的在前面. 排序以后,要合并了. 我自己的笨方法,说实在的问题真的很多.提交了好几次 ...