0

0

Linux如何使用epoll开发高并发服务_LinuxIO多路复用示例

冷漠man

冷漠man

发布时间:2025-11-29 18:38:02

|

912人浏览过

|

来源于php中文网

原创

答案:本文介绍了基于Linux epoll机制实现高性能回显服务器的方法,通过epoll_create、epoll_ctl和epoll_wait系统调用管理大量并发连接,结合非阻塞I/O与边缘触发模式提升效率,并给出完整C语言示例及关键优化点。

linux如何使用epoll开发高并发服务_linuxio多路复用示例

在开发高并发网络服务时,I/O多路复用是提升性能的核心技术之一。Linux下的 epoll 相比传统的 select 和 poll,具备更高的效率和可扩展性,特别适合处理成千上万的并发连接。本文通过一个简单的示例,说明如何使用 epoll 实现一个高性能的回显服务器(Echo Server)。

epoll 的基本原理

epoll 是 Linux 特有的 I/O 事件通知机制,它通过三个主要系统调用工作:

  • epoll_create:创建一个 epoll 实例。
  • epoll_ctl:向 epoll 实例注册、修改或删除文件描述符的监听事件。
  • epoll_wait:等待有事件发生的文件描述符,返回就绪列表。

epoll 使用红黑树管理监听的 fd,避免每次调用都传入全部 fd,同时通过就绪链表返回活跃连接,时间复杂度为 O(1),适合大规模并发场景。

实现一个简单的 epoll 回显服务器

下面是一个基于 epoll 的单线程 TCP 回显服务器的基本结构:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define MAX_EVENTS 1024
#define PORT 8888

// 设置文件描述符为非阻塞
int set_nonblocking(int fd) {
    int flags = fcntl(fd, F_GETFL, 0);
    if (flags == -1) flags = 0;
    return fcntl(fd, F_SETFL, flags | O_NONBLOCK);
}

int main() {
    int listen_sock, conn_sock, epoll_fd;
    struct sockaddr_in serv_addr, cli_addr;
    socklen_t cli_len = sizeof(cli_addr);
    struct epoll_event ev, events[MAX_EVENTS];

    // 创建监听 socket
    listen_sock = socket(AF_INET, SOCK_STREAM, 0);
    set_nonblocking(listen_sock);

    // 绑定地址和端口
    memset(&serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_addr.s_addr = INADDR_ANY;
    serv_addr.sin_port = htons(PORT);

    bind(listen_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr));
    listen(listen_sock, 10);

    // 创建 epoll 实例
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1) {
        perror("epoll_create1");
        return -1;
    }

    // 将监听 socket 加入 epoll,监听新连接
    ev.events = EPOLLIN;
    ev.data.fd = listen_sock;
    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, listen_sock, &ev) == -1) {
        perror("epoll_ctl: listen_sock");
        return -1;
    }

    printf("Server running on port %d\n", PORT);

    while (1) {
        int nfds = epoll_wait(epoll_fd, events, MAX_EVENTS, -1);
        if (nfds == -1) {
            perror("epoll_wait");
            break;
        }

        for (int i = 0; i < nfds; i++) {
            if (events[i].data.fd == listen_sock) {
                // 新连接到来
                conn_sock = accept(listen_sock, (struct sockaddr*)&cli_addr, &cli_len);
                if (conn_sock == -1) {
                    if (errno != EAGAIN && errno != EWOULDBLOCK)
                        perror("accept");
                    continue;
                }
                set_nonblocking(conn_sock);
                ev.events = EPOLLIN | EPOLLET;  // 边缘触发模式
                ev.data.fd = conn_sock;
                if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, conn_sock, &ev) == -1) {
                    perror("epoll_ctl: conn_sock");
                    close(conn_sock);
                }
            } else {
                // 已连接 socket 有数据可读
                conn_sock = events[i].data.fd;
                char buf[1024];
                ssize_t count;

                while ((count = read(conn_sock, buf, sizeof(buf))) > 0) {
                    write(conn_sock, buf, count); // 回显数据
                }

                if (count == 0) {
                    // 客户端关闭连接
                    close(conn_sock);
                    epoll_ctl(epoll_fd, EPOLL_CTL_DEL, conn_sock, NULL);
                } else if (count == -1) {
                    if (errno != EAGAIN)
                        close(conn_sock);
                }
            }
        }
    }

    close(listen_sock);
    close(epoll_fd);
    return 0;
}

关键点说明与优化建议

  • 非阻塞 I/O 必须配合 epoll 使用:所有加入 epoll 的 fd 都应设置为非阻塞,防止 read/write 阻塞整个事件循环。
  • 边缘触发(ET) vs 水平触发(LT):示例中使用了 EPOLLET 模式,只在状态变化时通知一次。使用 ET 时必须一次性处理完所有数据(如循环 read 直到 EAGAIN),否则可能丢失事件。
  • 错误处理不可忽略:read 返回 0 表示对端关闭,返回 -1 且 errno 为 EAGAIN 或 EWOULDBLOCK 表示无数据可读,不是错误。
  • 资源清理要及时:连接断开后应从 epoll 中删除,并关闭 fd,避免资源泄漏。

编译与测试

将代码保存为 echo_server.c,使用以下命令编译:

来福FM
来福FM

来福 - 你的私人AI电台

下载
gcc -o echo_server echo_server.c

运行服务器:

./echo_server

使用 telnet 或 nc 测试:

telnet 127.0.0.1 8888

输入任意内容,服务器会原样返回。

基本上就这些。掌握 epoll 的使用,是构建高性能网络服务的基础。虽然代码看起来简单,但背后涉及非阻塞 I/O、事件驱动、内存管理和并发控制等多个关键点。实际项目中可在此基础上引入线程池、缓冲区管理、协议解析等模块,逐步演进为完整的网络框架。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

387

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

611

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

351

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

256

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

597

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

522

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

639

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

599

2023.09.22

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

9

2026.01.14

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.1万人学习

Git 教程
Git 教程

共21课时 | 2.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号