Linux网络编程一步一步学-epoll同时处理海量连接的代码

本文介绍了一个使用Epoll实现的高效服务器程序,能同时处理数千个并发连接。程序通过Epoll机制监听socket上的读写事件,实现了非阻塞I/O模型。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <errno.h>
  4. #include <string.h>
  5. #include <sys/types.h>
  6. #include <netinet/in.h>
  7. #include <sys/socket.h>
  8. #include <sys/wait.h>
  9. #include <unistd.h>
  10. #include <arpa/inet.h>
  11. #include <openssl/ssl.h>
  12. #include <openssl/err.h>
  13. #include <fcntl.h>
  14. #include <sys/epoll.h>
  15. #include <sys/time.h>
  16. #include <sys/resource.h>
  17. #define MAXBUF 1024
  18. #define MAXEPOLLSIZE 10000
  19. /*
  20. setnonblocking - 设置句柄为非阻塞方式
  21. */
  22. int setnonblocking(int sockfd)
  23. {
  24.     if (fcntl(sockfd, F_SETFL, fcntl(sockfd, F_GETFD, 0)|O_NONBLOCK) == -1) {
  25.         return -1;
  26.     }
  27.     return 0;
  28. }
  29. /*
  30. handle_message - 处理每个 socket 上的消息收发
  31. */
  32. int handle_message(int new_fd)
  33. {
  34.     char buf[MAXBUF + 1];
  35.     int len;
  36.     /* 开始处理每个新连接上的数据收发 */
  37.     bzero(buf, MAXBUF + 1);
  38.     /* 接收客户端的消息 */
  39.     len = recv(new_fd, buf, MAXBUF, 0);
  40.     if (len > 0)
  41.         printf
  42.             ("%d接收消息成功:'%s',共%d个字节的数据/n",
  43.              new_fd, buf, len);
  44.     else {
  45.         if (len < 0)
  46.             printf
  47.                 ("消息接收失败!错误代码是%d,错误信息是'%s'/n",
  48.                  errno, strerror(errno));
  49.         close(new_fd);
  50.         return -1;
  51.     }
  52.     /* 处理每个新连接上的数据收发结束 */
  53.     return len;
  54. }
  55. /************关于本文档********************************************
  56. *filename: epoll-server.c
  57. *purpose: 演示epoll处理海量socket连接的方法
  58. *wrote by: zhoulifa(zhoulifa@163.com) 周立发(http://zhoulifa.bokee.com)
  59. Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言
  60. *date time:2007-01-31 21:00
  61. *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
  62. * 但请遵循GPL
  63. *Thanks to:Google
  64. *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
  65. * 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
  66. *********************************************************************/
  67. int main(int argc, char **argv)
  68. {
  69.     int listener, new_fd, kdpfd, nfds, n, ret, curfds;
  70.     socklen_t len;
  71.     struct sockaddr_in my_addr, their_addr;
  72.     unsigned int myport, lisnum;
  73.     struct epoll_event ev;
  74.     struct epoll_event events[MAXEPOLLSIZE];
  75.     struct rlimit rt;
  76.     if (argv[1])
  77.         myport = atoi(argv[1]);
  78.     else
  79.         myport = 7838;
  80.     if (argv[2])
  81.         lisnum = atoi(argv[2]);
  82.     else
  83.         lisnum = 2;
  84.     /* 设置每个进程允许打开的最大文件数 */
  85.     rt.rlim_max = rt.rlim_cur = MAXEPOLLSIZE;
  86.     if (setrlimit(RLIMIT_NOFILE, &rt) == -1) {
  87.         perror("setrlimit");
  88.         exit(1);
  89.     }
  90.     else printf("设置系统资源参数成功!/n");
  91.     /* 开启 socket 监听 */
  92.     if ((listener = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
  93.         perror("socket");
  94.         exit(1);
  95.     } else
  96.         printf("socket 创建成功!/n");
  97.     setnonblocking(listener);
  98.     bzero(&my_addr, sizeof(my_addr));
  99.     my_addr.sin_family = PF_INET;
  100.     my_addr.sin_port = htons(myport);
  101.     if (argv[3])
  102.         my_addr.sin_addr.s_addr = inet_addr(argv[3]);
  103.     else
  104.         my_addr.sin_addr.s_addr = INADDR_ANY;
  105.     if (bind
  106.         (listener, (struct sockaddr *) &my_addr, sizeof(struct sockaddr))
  107.         == -1) {
  108.         perror("bind");
  109.         exit(1);
  110.     } else
  111.         printf("IP 地址和端口绑定成功/n");
  112.     if (listen(listener, lisnum) == -1) {
  113.         perror("listen");
  114.         exit(1);
  115.     } else
  116.         printf("开启服务成功!/n");
  117.     /* 创建 epoll 句柄,把监听 socket 加入到 epoll 集合里 */
  118.     kdpfd = epoll_create(MAXEPOLLSIZE);
  119.     len = sizeof(struct sockaddr_in);
  120.     ev.events = EPOLLIN | EPOLLET;
  121.     ev.data.fd = listener;
  122.     if (epoll_ctl(kdpfd, EPOLL_CTL_ADD, listener, &ev) < 0) {
  123.         fprintf(stderr, "epoll set insertion error: fd=%d/n", listener);
  124.         return -1;
  125.     } else
  126.         printf("监听 socket 加入 epoll 成功!/n");
  127.     curfds = 1;
  128.     while (1) {
  129.         /* 等待有事件发生 */
  130.         nfds = epoll_wait(kdpfd, events, curfds, -1);
  131.         if (nfds == -1) {
  132.             perror("epoll_wait");
  133.             break;
  134.         }
  135.         /* 处理所有事件 */
  136.         for (n = 0; n < nfds; ++n) {
  137.             if (events[n].data.fd == listener) {
  138.                 new_fd = accept(listener, (struct sockaddr *) &their_addr,
  139.                                 &len);
  140.                 if (new_fd < 0) {
  141.                     perror("accept");
  142.                     continue;
  143.                 } else
  144.                     printf("有连接来自于: %d:%d, 分配的 socket 为:%d/n", inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port), new_fd);
  145.                 setnonblocking(new_fd);
  146.                 ev.events = EPOLLIN | EPOLLET;
  147.                 ev.data.fd = new_fd;
  148.                 if (epoll_ctl(kdpfd, EPOLL_CTL_ADD, new_fd, &ev) < 0) {
  149.                     fprintf(stderr, "把 socket '%d' 加入 epoll 失败!%s/n",
  150.                             new_fd, strerror(errno));
  151.                     return -1;
  152.                 }
  153.                 curfds++;
  154.             } else {
  155.                 ret = handle_message(events[n].data.fd);
  156.                 if (ret < 1 && errno != 11) {
  157.                     epoll_ctl(kdpfd, EPOLL_CTL_DEL, events[n].data.fd,
  158.                               &ev);
  159.                     curfds--;
  160.                 }
  161.             }
  162.         }
  163.     }
  164.     close(listener);
  165.     return 0;
  166. }
编译此程序用命令:
gcc -Wall epoll-server.c -o server

运行此程序需要具有管理员权限!

sudo ./server 7838 1

通过测试这一个服务器可能同时处理10000 -3 = 9997 个连接!

如果这是一个在线服务系统,那么它可以支持9997人同时在线,比如游戏、聊天等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值