网络编程:Boost.Asio实现跨平台的TCP服务器

这正是 Boost.Asio 最擅长的地方!它把 Linux 的 epoll 和 Windows 的 IOCP 封装成了统一的接口,让你完全不用写 #ifdef _WIN32 这种宏来区分平台。

下面我为你展示如何用 C++17 + Boost.Asio 实现一个标准的、跨平台的异步 TCP 服务器。

🛠️ 第一步:环境准备

在写代码之前,你需要确保安装了 Boost 库。

  • Ubuntu/Linux:

    bash 复制代码
    sudo apt-get install libboost-all-dev
  • Windows (使用 vcpkg):

    bash 复制代码
    vcpkg install boost-asio

💻 第二步:核心代码实现

这是一个基于 Reactor 模式 的服务器,支持高并发,且代码在 Windows 和 Linux 上完全通用。

cpp 复制代码
#include <iostream>
#include <vector>
#include <memory>
#include <thread>
#include <boost/asio.hpp>

using boost::asio::ip::tcp;
using namespace std;

// 1. 会话类:负责处理单个客户端的连接
class Session : public enable_shared_from_this<Session> {
    tcp::socket socket_;
    enum { max_length = 1024 };
    char data_[max_length];

public:
    Session(tcp::socket socket) : socket_(std::move(socket)) {}

    void start() {
        do_read();
    }

private:
    void do_read() {
        // 异步读取:不阻塞线程
        auto self(shared_from_this());
        socket_.async_read_some(
            boost::asio::buffer(data_, max_length),
            [this, self](boost::system::error_code ec, std::size_t length) {
                if (!ec) {
                    // 读取成功,异步回写
                    do_write(length);
                }
            });
    }

    void do_write(std::size_t length) {
        auto self(shared_from_this());
        boost::asio::async_write(
            socket_, boost::asio::buffer(data_, length),
            [this, self](boost::system::error_code ec, std::size_t /*length*/) {
                if (!ec) {
                    // 写完后继续读
                    do_read();
                }
            });
    }
};

// 2. 服务器类:负责监听端口和接受连接
class TcpServer {
    boost::asio::io_context& io_context_;
    tcp::acceptor acceptor_;

public:
    TcpServer(boost::asio::io_context& io_context, short port)
        : io_context_(io_context),
          acceptor_(io_context, tcp::endpoint(tcp::v4(), port))
    {
        do_accept();
    }

private:
    void do_accept() {
        // 异步接受新连接
        acceptor_.async_accept(
            [this](boost::system::error_code ec, tcp::socket socket) {
                if (!ec) {
                    // 创建会话对象并开始处理
                    make_shared<Session>(std::move(socket))->start();
                    cout << "New client connected" << endl;
                }
                // 继续接受下一个连接
                do_accept();
            });
    }
};

int main() {
    try {
        // 1. 创建 io_context (核心事件循环)
        boost::asio::io_context io_context;

        // 2. 启动服务器
        TcpServer server(io_context, 12345);

        cout << "Server running on port 12345..." << endl;

        // 3. 启动多线程池 (模拟多线程处理)
        // 在 Windows 下底层是 IOCP,Linux 下是 epoll
        vector<thread> threads;
        for (int i = 0; i < 4; ++i) {
            threads.emplace_back([&io_context]() {
                io_context.run();
            });
        }

        // 等待所有线程结束
        for (auto& t : threads) {
            t.join();
        }

    } catch (std::exception& e) {
        std::cerr << "Exception: " << e.what() << "\n";
    }

    return 0;
}

📝 第三步:编译与运行

Linux (g++)
bash 复制代码
g++ -std=c++17 server.cpp -o server -lboost_system -pthread
./server
Windows (CMake + vcpkg)

如果你使用 CMake,CMakeLists.txt 非常简单:

cmake 复制代码
cmake_minimum_required(VERSION 3.10)
project(TcpServer)

set(CMAKE_CXX_STANDARD 17)

find_package(Boost REQUIRED COMPONENTS system)

add_executable(server server.cpp)
target_link_libraries(server PRIVATE Boost::system)

🔍 为什么它能跨平台?(核心原理)

Boost.Asio 的魔法在于它封装了底层的差异:

层面 Linux 底层 Windows 底层 Boost.Asio 封装
I/O 多路复用 epoll IOCP (完成端口) boost::asio::io_context
Socket 句柄 int (文件描述符) SOCKET (句柄) boost::asio::ip::tcp::socket
错误码 errno WSAGetLastError() boost::system::error_code
启动/关闭 无特殊操作 WSAStartup / WSACleanup 自动处理 (无需手动调用)

🌟 代码亮点解析

  1. boost::asio::io_context

    这是整个库的心脏。在 Linux 上,调用 io_context.run() 时,它内部会去 epoll_wait;在 Windows 上,它会去 GetQueuedCompletionStatus (IOCP)。你完全不用关心底层。

  2. enable_shared_from_this

    Session 类中,我们使用了 shared_from_this()。这是为了防止异步操作(如 async_read)还在进行时,Session 对象被意外销毁。这是一种非常安全的内存管理方式。

  3. 多线程模型

    代码最后启动了 4 个线程运行 io_context.run()

    • Linux 下 :这 4 个线程会同时抢锁处理 epoll 返回的就绪事件。
    • Windows 下:这 4 个线程会作为 IOCP 的工作线程池,自动负载均衡。

🚀 总结

使用 Boost.Asio,你只需要关注业务逻辑 (怎么读、怎么写),而不用关心系统调用(怎么监听、怎么轮询)。这就是它成为 C++ 网络编程事实标准的原因。

相关推荐
2301_809051143 小时前
Linux 网络编程 学习笔记
linux·网络·学习
wanhengidc3 小时前
服务器租用有何优点
运维·服务器·安全·web安全
坤昱4 小时前
cfs调度类深入解刨——最新内核细节分析2
linux·服务器·cfs·cfs调度·eevdf调度·eevdf·kernel 7.1
艾莉丝努力练剑4 小时前
【Linux:文件】Ext系列文件系统进阶
linux·运维·服务器·c++·文件系统·文件io·ext
海市公约4 小时前
Linux核心基础命令与权限管理实战指南
linux·运维·服务器·vim·权限管理·系统监控·命令行
mixboot5 小时前
Linux 进程工作目录查看利器:pwdx 命令详解
linux·运维·服务器
星恒讯工业路由器6 小时前
Wi‑Fi DCM 双载波调制解析
网络·信息与通信·wifi7·wifi6·wi‑fi dcm 双载波调制
IP搭子来一个7 小时前
爬虫采集大量返回 403、429,到底卡在哪一环?
网络·爬虫·python
之歆7 小时前
Day16_JavaScript 轮播图与事件工程实战(下篇)
服务器·开发语言·前端·javascript·网络·性能优化
IT大白鼠7 小时前
ICMP协议详解:从基础原理到网络应用实践
网络