本章主要简述epoll的基础理论知识。方便理解libevent。

2.1 流?I/O操作/阻塞

2.1.1 流

  • 可以进行I/O操作的内核对象
  • 文件、管道、套接字……
  • 流的入口:文件描述符(fd)

2.1.2 I/O操作

libevent-2-io操作01.png


libevent-2-io操作02.png


所有对流的读写操作,我们都可以称之为IO操作。

那么当一个流中再没有数据,read的时候,或者说在流中已经写满了数据,再write,我们的IO操作就会出现一种现象,就是阻塞现象。

2.1.2 阻塞

libevent-2-阻塞01.png


libevent-2-阻塞02.png


  • 阻塞等待

空出大脑可以安心睡觉。(不占用CPU宝贵的时间片)

  • 非阻塞,忙轮询

浪费时间,浪费电话费,占用快递员时间(占用CPU,系统资源)

2.2 解决阻塞死等待的办法

2.2.1 阻塞死等待的缺点libevent-2-解决阻塞01.png

2.2.2 办法一:非阻塞、忙轮询

libevent-2-解决阻塞02.png

  1. while true {
  2. for i in 流[] {
  3. if i has 数据 {
  4. 或者 其他处理
  5. }
  6. }
  7. }

2.2.3 办法二:select

libevent-2-解决阻塞03.png

2.2.3 办法三:epoll

libevent-2-解决阻塞04.png

  1. while true {
  2. 可处理的流[] = epoll_wait(epoll_fd); //阻塞
  3. for i in 可处理的流[] {
  4. 或者 其他处理
  5. }
  6. }

2.3 什么是epoll

  • 与select,poll一样,对I/O多路复用的技术
  • 只关心“活跃”的链接,无需遍历全部描述符集合
  • 能够处理大量的链接请求(系统可以打开的文件数目)

2.4 epoll API

2.4.1 创建EPOLL

  1. /**
  2. * @param size 告诉内核监听的数目
  3. *
  4. * @returns 返回一个epoll句柄(即一个文件描述符)
  5. */
  6. int epoll_create(int size);
  1. int epfd = epoll_create(1000);

libevent-2-epoll-api01.png

2.4.2 控制EPOLL

  1. /**
  2. * @param epfd 用epoll_create所创建的epoll句柄
  3. * @param op 表示对epoll监控描述符控制的动作
  4. *
  5. * EPOLL_CTL_ADD(注册新的fd到epfd)
  6. * EPOLL_CTL_MOD(修改已经注册的fd的监听事件)
  7. * EPOLL_CTL_DEL(epfd删除一个fd)
  8. *
  9. * @param fd 需要监听的文件描述符
  10. * @param event 告诉内核需要监听的事件
  11. *
  12. * @returns 成功返回0,失败返回-1, errno查看错误信息
  13. */
  14. int epoll_ctl(int epfd, int op, int fd,
  15. struct epoll_event *event);
  16. struct epoll_event {
  17. __uint32_t events; /* epoll 事件 */
  18. epoll_data_t data; /* 用户传递的数据 */
  19. }
  20. /*
  21. * events : {EPOLLIN, EPOLLOUT, EPOLLPRI,
  22. EPOLLHUP, EPOLLET, EPOLLONESHOT}
  23. */
  24. typedef union epoll_data {
  25. void *ptr;
  26. int fd;
  27. uint32_t u32;
  28. uint64_t u64;
  29. } epoll_data_t;
  1. struct epoll_event new_event;
  2. new_event.events = EPOLLIN | EPOLLOUT;
  3. new_event.data.fd = 5;
  4. epoll_ctl(epfd, EPOLL_CTL_ADD, 5, &new_event);

libevent-2-epoll-api02.png

2.4.3 等待EPOLL

  1. /**
  2. *
  3. * @param epfd 用epoll_create所创建的epoll句柄
  4. * @param event 从内核得到的事件集合
  5. * @param maxevents 告知内核这个events有多大,
  6. * 注意: 值 不能大于创建epoll_create()时的size.
  7. * @param timeout 超时时间
  8. * -1: 永久阻塞
  9. * 0: 立即返回,非阻塞
  10. * >0: 指定微秒
  11. *
  12. * @returns 成功: 有多少文件描述符就绪,时间到时返回0
  13. * 失败: -1, errno 查看错误
  14. */
  15. int epoll_wait(int epfd, struct epoll_event *event,
  16. int maxevents, int timeout);
  1. struct epoll_event my_event[1000];
  2. int event_cnt = epoll_wait(epfd, my_event, 1000, -1);

libevent-2-epoll-api03.png

2.4.4 epoll编程框架

  1. //创建 epoll
  2. int epfd = epoll_crete(1000);
  3. //将 listen_fd 添加进 epoll 中
  4. epoll_ctl(epfd, EPOLL_CTL_ADD, listen_fd,&listen_event);
  5. while (1) {
  6. //阻塞等待 epoll 中 的fd 触发
  7. int active_cnt = epoll_wait(epfd, events, 1000, -1);
  8. for (i = 0 ; i < active_cnt; i++) {
  9. if (evnets[i].data.fd == listen_fd) {
  10. //accept. 并且将新accept 的fd 加进epoll中.
  11. }
  12. else if (events[i].events & EPOLLIN) {
  13. //对此fd 进行读操作
  14. }
  15. else if (events[i].events & EPOLLOUT) {
  16. //对此fd 进行写操作
  17. }
  18. }
  19. }

2.5 触发模式

2.5.1 水平触发

libevent-2-epoll触发模式01.png


libevent-2-epoll触发模式02.png


水平触发的主要特点是,如果用户在监听epoll事件,当内核有事件的时候,会拷贝给用户态事件,但是如果用户只处理了一次,那么剩下没有处理的会在下一次epoll_wait再次返回该事件。

这样如果用户永远不处理这个事件,就导致每次都会有该事件从内核到用户的拷贝,耗费性能,但是水平触发相对安全,最起码事件不会丢掉,除非用户处理完毕。

2.5.2 边缘触发

libevent-2-epoll触发模式03.png


libevent-2-epoll触发模式04.png

边缘触发,相对跟水平触发相反,当内核有事件到达, 只会通知用户一次,至于用户处理还是不处理,以后将不会再通知。这样减少了拷贝过程,增加了性能,但是相对来说,如果用户马虎忘记处理,将会产生事件丢的情况。

2.6 epoll服务器

2.6.1 服务端

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <ctype.h>
  4. #include <string.h>
  5. #include <unistd.h>
  6. #include <sys/types.h>
  7. #include <sys/socket.h>
  8. #include <arpa/inet.h>
  9. #include <sys/epoll.h>
  10. #define SERVER_PORT (7778)
  11. #define EPOLL_MAX_NUM (2048)
  12. #define BUFFER_MAX_LEN (4096)
  13. char buffer[BUFFER_MAX_LEN];
  14. void str_toupper(char *str)
  15. {
  16. int i;
  17. for (i = 0; i < strlen(str); i ++) {
  18. str[i] = toupper(str[i]);
  19. }
  20. }
  21. int main(int argc, char **argv)
  22. {
  23. int listen_fd = 0;
  24. int client_fd = 0;
  25. struct sockaddr_in server_addr;
  26. struct sockaddr_in client_addr;
  27. socklen_t client_len;
  28. int epfd = 0;
  29. struct epoll_event event, *my_events;
  30. // socket
  31. listen_fd = socket(AF_INET, SOCK_STREAM, 0);
  32. // bind
  33. server_addr.sin_family = AF_INET;
  34. server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
  35. server_addr.sin_port = htons(SERVER_PORT);
  36. bind(listen_fd, (struct sockaddr*)&server_addr, sizeof(server_addr));
  37. // listen
  38. listen(listen_fd, 10);
  39. // epoll create
  40. epfd = epoll_create(EPOLL_MAX_NUM);
  41. if (epfd < 0) {
  42. perror("epoll create");
  43. goto END;
  44. }
  45. // listen_fd -> epoll
  46. event.events = EPOLLIN;
  47. event.data.fd = listen_fd;
  48. if (epoll_ctl(epfd, EPOLL_CTL_ADD, listen_fd, &event) < 0) {
  49. perror("epoll ctl add listen_fd ");
  50. goto END;
  51. }
  52. my_events = malloc(sizeof(struct epoll_event) * EPOLL_MAX_NUM);
  53. while (1) {
  54. // epoll wait
  55. int active_fds_cnt = epoll_wait(epfd, my_events, EPOLL_MAX_NUM, -1);
  56. int i = 0;
  57. for (i = 0; i < active_fds_cnt; i++) {
  58. // if fd == listen_fd
  59. if (my_events[i].data.fd == listen_fd) {
  60. //accept
  61. client_fd = accept(listen_fd, (struct sockaddr*)&client_addr, &client_len);
  62. if (client_fd < 0) {
  63. perror("accept");
  64. continue;
  65. }
  66. char ip[20];
  67. printf("new connection[%s:%d]\n", inet_ntop(AF_INET, &client_addr.sin_addr, ip, sizeof(ip)), ntohs(client_addr.sin_port));
  68. event.events = EPOLLIN | EPOLLET;
  69. event.data.fd = client_fd;
  70. epoll_ctl(epfd, EPOLL_CTL_ADD, client_fd, &event);
  71. }
  72. else if (my_events[i].events & EPOLLIN) {
  73. printf("EPOLLIN\n");
  74. client_fd = my_events[i].data.fd;
  75. // do read
  76. buffer[0] = '\0';
  77. int n = read(client_fd, buffer, 5);
  78. if (n < 0) {
  79. perror("read");
  80. continue;
  81. }
  82. else if (n == 0) {
  83. epoll_ctl(epfd, EPOLL_CTL_DEL, client_fd, &event);
  84. close(client_fd);
  85. }
  86. else {
  87. printf("[read]: %s\n", buffer);
  88. buffer[n] = '\0';
  89. #if 1
  90. str_toupper(buffer);
  91. write(client_fd, buffer, strlen(buffer));
  92. printf("[write]: %s\n", buffer);
  93. memset(buffer, 0, BUFFER_MAX_LEN);
  94. #endif
  95. /*
  96. event.events = EPOLLOUT;
  97. event.data.fd = client_fd;
  98. epoll_ctl(epfd, EPOLL_CTL_MOD, client_fd, &event);
  99. */
  100. }
  101. }
  102. else if (my_events[i].events & EPOLLOUT) {
  103. printf("EPOLLOUT\n");
  104. /*
  105. client_fd = my_events[i].data.fd;
  106. str_toupper(buffer);
  107. write(client_fd, buffer, strlen(buffer));
  108. printf("[write]: %s\n", buffer);
  109. memset(buffer, 0, BUFFER_MAX_LEN);
  110. event.events = EPOLLIN;
  111. event.data.fd = client_fd;
  112. epoll_ctl(epfd, EPOLL_CTL_MOD, client_fd, &event);
  113. */
  114. }
  115. }
  116. }
  117. END:
  118. close(epfd);
  119. close(listen_fd);
  120. return 0;
  121. }

2.6.2 客户端

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <strings.h>
  5. #include <sys/types.h>
  6. #include <sys/socket.h>
  7. #include <arpa/inet.h>
  8. #include <unistd.h>
  9. #include <fcntl.h>
  10. #define MAX_LINE (1024)
  11. #define SERVER_PORT (7778)
  12. void setnoblocking(int fd)
  13. {
  14. int opts = 0;
  15. opts = fcntl(fd, F_GETFL);
  16. opts = opts | O_NONBLOCK;
  17. fcntl(fd, F_SETFL);
  18. }
  19. int main(int argc, char **argv)
  20. {
  21. int sockfd;
  22. char recvline[MAX_LINE + 1] = {0};
  23. struct sockaddr_in server_addr;
  24. if (argc != 2) {
  25. fprintf(stderr, "usage ./client <SERVER_IP>\n");
  26. exit(0);
  27. }
  28. // 创建socket
  29. if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
  30. fprintf(stderr, "socket error");
  31. exit(0);
  32. }
  33. // server addr 赋值
  34. bzero(&server_addr, sizeof(server_addr));
  35. server_addr.sin_family = AF_INET;
  36. server_addr.sin_port = htons(SERVER_PORT);
  37. if (inet_pton(AF_INET, argv[1], &server_addr.sin_addr) <= 0) {
  38. fprintf(stderr, "inet_pton error for %s", argv[1]);
  39. exit(0);
  40. }
  41. // 链接服务端
  42. if (connect(sockfd, (struct sockaddr*) &server_addr, sizeof(server_addr)) < 0) {
  43. perror("connect");
  44. fprintf(stderr, "connect error\n");
  45. exit(0);
  46. }
  47. setnoblocking(sockfd);
  48. char input[100];
  49. int n = 0;
  50. int count = 0;
  51. // 不断的从标准输入字符串
  52. while (fgets(input, 100, stdin) != NULL)
  53. {
  54. printf("[send] %s\n", input);
  55. n = 0;
  56. // 把输入的字符串发送 到 服务器中去
  57. n = send(sockfd, input, strlen(input), 0);
  58. if (n < 0) {
  59. perror("send");
  60. }
  61. n = 0;
  62. count = 0;
  63. // 读取 服务器返回的数据
  64. while (1)
  65. {
  66. n = read(sockfd, recvline + count, MAX_LINE);
  67. if (n == MAX_LINE)
  68. {
  69. count += n;
  70. continue;
  71. }
  72. else if (n < 0){
  73. perror("recv");
  74. break;
  75. }
  76. else {
  77. count += n;
  78. recvline[count] = '\0';
  79. printf("[recv] %s\n", recvline);
  80. break;
  81. }
  82. }
  83. }
  84. return 0;
  85. }