0%

1_2_asio_理论基础

简介

  • BOOST.Asio 网络服务相关模块的笔记

Boost.Asio 库 详解

Boost.Asio 库详解

Boost.Asio 是一个功能强大的 C++ 网络和并发编程库,属于 Boost 库的一部分。它提供了跨平台的异步 I/O 支持,能够处理网络通信、定时器和文件 I/O 等任务。以下是对 Boost.Asio 的详细解析:


1. 核心概念

I/O 服务 (io_context)

  • 核心机制boost::asio::io_context 是 Asio 的核心,用于调度所有异步操作。
    它管理 I/O 操作队列并为异步事件提供执行环境。
  • 功能
    • 注册 I/O 操作。
    • 调度任务。
    • 运行事件循环。

工作线程

  • 多个线程可以同时调用 io_context.run() 来处理任务队列,实现多线程并发。

异步操作

  • Boost.Asio 强调异步操作,以避免阻塞。
  • 使用回调函数或协程处理完成的任务。

Handlers 和 Executors

  • Handlers:异步操作完成后调用的回调函数。
  • Executors:控制任务在哪个上下文中执行(单线程、多线程等)。

2. 常见功能模块

网络操作

Boost.Asio 提供了对常见网络协议(TCP、UDP)的支持。

TCP 通信
  1. TCP Server 示例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    #include <boost/asio.hpp>
    #include <iostream>

    using boost::asio::ip::tcp;

    int main() {
    boost::asio::io_context io_context;

    // 创建一个监听器,绑定到 127.0.0.1:1234
    tcp::acceptor acceptor(io_context, tcp::endpoint(tcp::v4(), 1234));

    // 等待客户端连接
    tcp::socket socket(io_context);
    acceptor.accept(socket);

    // 发送欢迎消息
    std::string message = "Hello from server!";
    boost::asio::write(socket, boost::asio::buffer(message));

    return 0;
    }
  2. TCP Client 示例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    #include <boost/asio.hpp>
    #include <iostream>

    using boost::asio::ip::tcp;

    int main() {
    boost::asio::io_context io_context;

    // 创建 TCP 连接到 127.0.0.1:1234
    tcp::resolver resolver(io_context);
    auto endpoints = resolver.resolve("127.0.0.1", "1234");
    tcp::socket socket(io_context);
    boost::asio::connect(socket, endpoints);

    // 接收消息
    char buffer[1024];
    size_t length = socket.read_some(boost::asio::buffer(buffer));
    std::cout << "Message from server: " << std::string(buffer, length) << "\n";

    return 0;
    }
UDP 通信
  1. UDP Server 示例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    #include <boost/asio.hpp>
    #include <iostream>

    using boost::asio::ip::udp;

    int main() {
    boost::asio::io_context io_context;

    udp::socket socket(io_context, udp::endpoint(udp::v4(), 1234));
    char data[1024];
    udp::endpoint sender_endpoint;

    // 接收数据
    size_t length = socket.receive_from(boost::asio::buffer(data), sender_endpoint);
    std::cout << "Received: " << std::string(data, length) << "\n";

    // 发送响应
    std::string response = "Hello from server!";
    socket.send_to(boost::asio::buffer(response), sender_endpoint);

    return 0;
    }
  2. UDP Client 示例

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    #include <boost/asio.hpp>
    #include <iostream>

    using boost::asio::ip::udp;

    int main() {
    boost::asio::io_context io_context;

    udp::resolver resolver(io_context);
    auto endpoints = resolver.resolve(udp::v4(), "127.0.0.1", "1234");

    udp::socket socket(io_context);
    socket.open(udp::v4());

    // 发送数据
    std::string message = "Hello from client!";
    socket.send_to(boost::asio::buffer(message), *endpoints.begin());

    // 接收响应
    char buffer[1024];
    udp::endpoint server_endpoint;
    size_t length = socket.receive_from(boost::asio::buffer(buffer), server_endpoint);
    std::cout << "Response from server: " << std::string(buffer, length) << "\n";

    return 0;
    }

定时器

Boost.Asio 提供了基于时间的功能,例如超时处理或定时任务。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
#include <boost/asio.hpp>
#include <iostream>

void print(const boost::system::error_code&) {
std::cout << "Timer expired!\n";
}

int main() {
boost::asio::io_context io_context;

// 创建定时器,设置 5 秒超时
boost::asio::steady_timer timer(io_context, boost::asio::chrono::seconds(5));
timer.async_wait(&print);

// 启动事件循环
io_context.run();

return 0;
}

3. 同步与异步模式

同步模式

  • 直接调用函数(如 boost::asio::read()),操作完成后返回。
  • 简单易用,但可能导致线程阻塞。

异步模式

  • 通过 async_* 系列函数注册回调函数(如 boost::asio::async_read())。
  • 回调函数在操作完成后被触发。
  • 非阻塞,更适合高性能和并发需求。

4. 常见特性

Error Handling

  • 使用 boost::system::error_code 对象捕获错误。
  • 示例:
    1
    2
    3
    4
    5
    boost::system::error_code ec;
    boost::asio::write(socket, boost::asio::buffer(data), ec);
    if (ec) {
    std::cerr << "Error: " << ec.message() << "\n";
    }

多线程支持

  • 多线程可通过共享一个 io_context 对象实现并发任务调度。
  • 示例:
    1
    2
    3
    4
    5
    6
    7
    boost::asio::io_context io_context;

    std::thread t1([&io_context]() { io_context.run(); });
    std::thread t2([&io_context]() { io_context.run(); });

    t1.join();
    t2.join();

5. Boost.Asio 的优点

  1. 跨平台:支持多种操作系统和编译器。
  2. 高性能:高效的异步 I/O 模型。
  3. 易于扩展:支持自定义 I/O 对象和扩展功能。
  4. 与标准库兼容:可以结合 std::future 和 C++20 协程使用。

6. Boost.Asio 与 C++ 标准库

Boost.Asio 的一些功能已被纳入 C++ 标准库,如 C++11 的 std::thread 和 C++20 的网络库(<experimental/net>)。如果项目需要未来的标准兼容性,可以考虑迁移到标准库的实现。

BOOST Asio模块 是什么 详解

Boost.Asio是Boost库中的一个模块,提供了跨平台的异步I/O和网络编程功能。它被设计用于简化异步编程、网络编程和低级I/O操作,使开发者能够更轻松地创建高性能的网络应用程序。

主要特点和用途包括:

  1. 异步I/O操作支持:Boost.Asio提供了异步I/O操作的框架,允许开发者执行非阻塞的I/O操作,以便同时处理多个操作,提高程序的并发性能。

  2. 跨平台性:它可以在多种操作系统上运行,包括Windows、Linux、macOS等,提供了对不同平台的统一接口,简化了跨平台开发。

  3. 网络编程:Boost.Asio支持TCP、UDP、SSL等网络协议,允许开发者轻松地创建网络应用程序,例如服务器、客户端和中间件。

  4. IO对象和Handler:使用Boost.Asio时,I/O操作被封装在I/O对象中,通过异步操作和回调函数(handler)来处理操作完成的通知。

  5. 定时器和异步任务:它提供了定时器和异步任务的功能,允许开发者进行计时操作和调度异步任务。

  6. 高性能:由于采用异步的方式处理I/O操作,Boost.Asio可以实现高性能的网络应用程序。

下面是一个简单的示例,展示了使用Boost.Asio库进行异步操作的方式:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#include <iostream>
#include <boost/asio.hpp>

void print(const boost::system::error_code& /*ec*/) {
std::cout << "Hello, World!" << std::endl;
}

int main() {
boost::asio::io_service io_service;

boost::asio::steady_timer timer(io_service, boost::asio::chrono::seconds(1));
timer.async_wait(&print);

io_service.run();

return 0;
}

这个示例创建了一个定时器,在1秒后触发一个异步操作,调用print函数输出”Hello, World!”。io_service.run()会一直运行,直到所有的异步操作完成或者io_service被停止。

Boost.Asio是一个强大而灵活的库,可以用于构建各种类型的网络应用程序,但它的使用需要一定的学习曲线。对于更复杂的应用程序,可能需要深入学习Boost.Asio文档并熟悉其API和异步编程的概念。

boost::asio::io_context 详解

boost::asio::io_context 是 Boost.Asio 库中的关键类之一,用于管理异步 I/O 操作和事件处理循环。它是 Boost.Asio 提供的核心组件之一,用于实现异步编程模型。

以下是对 boost::asio::io_context 的详细解释:

  1. 异步 I/O 和事件循环
    boost::asio::io_context 提供了一个事件处理循环,用于管理异步 I/O 操作。它负责处理套接字、定时器、异步任务等操作的调度和执行。

  2. 多线程支持
    io_context 可以在多个线程中共享使用,允许在多线程环境中处理异步操作。多个线程可以同时运行 io_context::run(),从而实现并发的异步操作处理。

  3. 操作队列管理
    它维护一个操作队列,用于存储待处理的异步操作。当调用 io_context::run() 时,它会处理队列中的操作,直到队列为空或者调用 io_context::stop() 来停止事件处理循环。

  4. 定时器支持
    io_context 可以创建定时器,用于在指定的时间点执行操作或者在指定的时间间隔内重复执行操作。

  5. 回调机制
    当异步操作完成或者发生错误时,io_context 会调用相关的回调函数,通知调用者操作的完成情况。这样可以实现非阻塞的异步操作处理。

  6. 资源管理
    io_context 负责管理和分配底层资源,如套接字和文件句柄等资源,在操作完成后进行资源的释放。

以下是一个简单的示例,展示了如何创建和使用 boost::asio::io_context

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
#include <iostream>
#include <boost/asio.hpp>

int main() {
boost::asio::io_context io_context;

// 异步操作,打印 "Hello, World!" 后停止 io_context
io_context.post([]() {
std::cout << "Hello, World!" << std::endl;
});

// 开始处理异步操作,直到所有操作完成
io_context.run();

return 0;
}

这个简单的示例创建了一个 io_context 对象,并使用 post() 方法在 io_context 上投递了一个异步操作,打印 “Hello, World!”。然后调用 io_context.run() 来启动事件循环,处理投递的异步操作,直到操作完成。在这个例子中,异步操作完成后程序退出。

boost::asio::io_context 是 Boost.Asio 中重要的核心组件之一,是实现异步 I/O 编程的关键部分。通过它,可以实现高效的异步操作处理和事件驱动的网络编程。

boost::asio::thread_pool 详解

boost::asio::thread_pool 是 Boost.Asio 库中的一个类,用于管理线程池,方便地处理异步操作。它提供了一种机制,允许在一个固定数量的线程池中执行异步任务。

以下是对 boost::asio::thread_pool 的详细解释:

  1. 线程池管理
    boost::asio::thread_pool 封装了一个线程池,可以在其中执行异步任务。它负责管理一组线程,这些线程可以被异步操作共享使用。

  2. 简化异步任务处理
    通过 thread_pool,可以轻松地管理异步操作的执行。无需手动创建和管理线程,而是将异步任务提交到线程池中,由线程池自动调度和执行。

  3. 控制线程数量
    可以在创建 thread_pool 时指定线程的数量。这个数量通常取决于系统资源和任务需求,可以避免创建过多线程而导致资源浪费。

  4. 异步任务调度
    thread_pool 提供了一种方便的方式来调度异步任务,使得多个任务可以并发执行,提高程序的并发性能。

  5. 资源管理
    线程池负责管理其内部线程的资源分配和释放,当不再需要时,可以正确地释放线程资源。

以下是一个简单的示例,展示了如何使用 boost::asio::thread_pool

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
#include <iostream>
#include <boost/asio.hpp>

int main() {
boost::asio::io_context io_context;
boost::asio::thread_pool thread_pool(4); // 创建拥有4个线程的线程池

// 向线程池提交异步任务
for (int i = 0; i < 8; ++i) {
boost::asio::post(thread_pool, [i]() {
std::cout << "Task " << i << " executed in thread: " << std::this_thread::get_id() << std::endl;
});
}

// 等待所有任务完成
thread_pool.join();

return 0;
}

在这个示例中,首先创建了一个 boost::asio::io_context 对象和一个拥有4个线程的 boost::asio::thread_pool 对象。然后使用 boost::asio::post() 将8个异步任务提交到线程池中执行。每个任务输出了自己的标识符和线程ID。最后,通过 thread_pool.join() 等待所有任务完成。

boost::asio::thread_pool 提供了一种便捷的方式来管理并发任务,避免了手动管理线程的复杂性,使得异步任务的管理更加简单和高效。

boost::asio::ip::tcp::socket 详解

boost::asio::ip::tcp::socket 是 Boost.Asio 库中用于 TCP 协议通信的关键类之一。它提供了在 C++ 中进行 TCP 连接的接口,并允许进行异步的数据传输。

以下是对 boost::asio::ip::tcp::socket 的详细解释:

  1. TCP 套接字
    boost::asio::ip::tcp::socket 是用于 TCP 协议通信的套接字类。通过它可以进行基于流的数据传输。

  2. 异步 I/O
    tcp::socket 支持异步 I/O 操作,可以使用异步读取和写入函数来进行非阻塞的数据传输,从而允许在进行网络通信时不阻塞当前线程。

  3. 连接管理
    通过 tcp::socket 可以连接到远程服务器,也可以接受远程客户端的连接请求,因此它可以用于客户端和服务器端编程。

  4. 数据传输
    tcp::socket 提供了方法来发送和接收数据。可以使用 async_read_some()async_write_some() 等函数进行异步数据读取和写入。

  5. 错误处理
    当发生错误时,tcp::socket 会生成相应的错误码,开发者可以根据错误码来处理异常情况。

  6. 关闭连接
    可以通过 close() 方法关闭套接字,释放资源并关闭与远程端点的连接。

以下是一个简单的示例,展示了如何使用 boost::asio::ip::tcp::socket 进行简单的 TCP 通信:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include <iostream>
#include <boost/asio.hpp>

using namespace boost::asio;
using namespace boost::asio::ip;

int main() {
io_context io_context;

// 创建套接字
tcp::socket socket(io_context);

try {
// 连接到服务器
tcp::endpoint endpoint(address::from_string("127.0.0.1"), 8080);
socket.connect(endpoint);

// 发送数据
std::string message = "Hello, Server!";
socket.write_some(buffer(message));

// 读取服务器响应
char data[128];
size_t len = socket.read_some(buffer(data, 128));
std::cout << "Received: " << std::string(data, len) << std::endl;

// 关闭套接字
socket.close();
} catch (const std::exception& e) {
std::cerr << "Error: " << e.what() << std::endl;
}

return 0;
}

这个简单的示例创建了一个 TCP 套接字 tcp::socket,连接到本地 IP 地址 127.0.0.1 的端口 8080,发送 “Hello, Server!” 消息,并等待服务器的响应。这个示例只是一个基本的演示,实际中可能需要更复杂的错误处理和异步操作。

boost::asio::ip::tcp::acceptor 详解

boost::asio::ip::tcp::acceptor 是 Boost 库中 Asio 网络编程库的一部分,用于在 C++ 中实现基于 TCP 协议的异步接受连接的操作。

1. boost::asio::ip::tcp::acceptor 概述

  • 作用: acceptor 用于在服务器端监听指定的 IP 地址和端口,并接受传入的连接请求。
  • 工作原理: acceptor 负责等待来自客户端的连接请求,并在连接请求到达时接受连接。它允许服务器接受传入连接,并将其传递给对应的处理器。
  • 使用场景: 在服务器端实现 TCP 服务时,acceptor 通常与 socket 结合使用,通过 acceptor 接受连接,并创建一个新的 socket 用于处理客户端和服务器之间的通信。

2. boost::asio::ip::tcp::acceptor 的基本用法

以下是一个简单的示例,演示如何使用 acceptor

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#include <boost/asio.hpp>
#include <iostream>

void handle_accept(const boost::system::error_code& error) {
if (!error) {
// 处理新连接
} else {
// 处理连接错误
}
}

int main() {
boost::asio::io_context io_context;
boost::asio::ip::tcp::acceptor acceptor(io_context);

// 在本地地址 0.0.0.0,端口号 8080 上监听连接
boost::asio::ip::tcp::endpoint endpoint(boost::asio::ip::tcp::v4(), 8080);
acceptor.open(endpoint.protocol());
acceptor.bind(endpoint);
acceptor.listen();

// 异步等待连接请求
acceptor.async_accept(
[&acceptor](const boost::system::error_code& error, boost::asio::ip::tcp::socket socket) {
handle_accept(error);
}
);

io_context.run();

return 0;
}

3. 重要方法和功能

  • open() 打开接受器并准备接受连接。
  • bind() 将接受器绑定到指定的 IP 地址和端口。
  • listen() 开始监听传入的连接请求。
  • async_accept() 异步接受传入的连接请求。当有新的连接请求到达时,调用回调函数处理连接。

4. 注意事项

  • 使用 async_accept 进行异步连接处理,必须确保在回调函数中处理连接,否则连接将被丢弃。
  • acceptor 通常与 io_context 一起使用,后者用于异步操作的事件循环处理。

请根据具体的需求和网络应用程序的逻辑,结合 Boost.Asio 的其他部分,例如 socket 类等,来构建一个完整的网络应用程序。

boost::asio::ip::tcp::acceptor 初始化

在使用 boost::asio::ip::tcp::acceptor 初始化之前,需要提供一个 io_context 对象以及一个 endpoint(端点)来指定监听的地址和端口。

下面是一个简单的示例,演示了如何初始化 boost::asio::ip::tcp::acceptor

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
#include <boost/asio.hpp>
#include <iostream>

int main() {
// 创建 io_context 对象
boost::asio::io_context io_context;

// 创建 acceptor 对象
boost::asio::ip::tcp::acceptor acceptor(io_context);

try {
// 定义 endpoint(端点),指定要监听的 IP 地址和端口
boost::asio::ip::tcp::endpoint endpoint(boost::asio::ip::tcp::v4(), 8080);

// 打开 acceptor,并绑定到指定端点
acceptor.open(endpoint.protocol());
acceptor.bind(endpoint);

// 开始监听传入的连接
acceptor.listen();

std::cout << "Server started and listening on port 8080" << std::endl;

// 在这之后,可以使用 acceptor 的 async_accept() 方法来接受连接请求并处理连接
// acceptor.async_accept(...);

io_context.run(); // 启动 IO 上下文的事件循环
} catch (std::exception& e) {
std::cerr << "Exception: " << e.what() << std::endl;
}

return 0;
}

在这个示例中:

  1. 创建了一个 io_context 对象用于异步 I/O 操作的事件循环。
  2. 使用 boost::asio::ip::tcp::acceptor 创建了一个接受器对象。
  3. 指定了一个 endpoint,即要监听的 IP 地址和端口。在此示例中,IP 地址是 IPv4 的地址,端口是 8080。
  4. 打开 acceptor 并将其绑定到指定的端点。
  5. 调用 listen() 开始监听传入的连接请求。
  6. io_context.run() 启动了异步操作的事件循环,以便接受来自客户端的连接请求。

请注意,这只是初始化 boost::asio::ip::tcp::acceptor 的一个基本示例。实际的应用程序需要根据需求进一步扩展,可能需要添加异步连接接受处理等操作。

boost::asio::ip::tcp::endpoint 详解

boost::asio::ip::tcp::endpoint 是 Boost.Asio 中用于表示 TCP 协议的端点(endpoint)的类。它定义了一个包含 IP 地址和端口号的组合,用于标识网络上的一个节点。

主要特点和功能:

  1. 表示网络上的节点: tcp::endpoint 表示了网络上的一个节点,它由 IP 地址和端口号组成。
  2. 跨平台性: endpoint 类支持跨平台操作,可以在各种操作系统上使用。
  3. 灵活性: 允许用户指定 IPv4 或 IPv6 地址,以及端口号。

主要方法和操作:

  • 构造函数: 提供了多种构造函数来创建 tcp::endpoint 实例,可以根据不同的参数设置 IP 地址和端口号。
  • 协议访问: 可以通过 protocol() 方法访问端点的协议信息。
  • IP 地址和端口访问: 通过 address()port() 方法获取端点的 IP 地址和端口号。

示例用法:

以下是一个示例,演示了如何使用 tcp::endpoint

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <boost/asio.hpp>
#include <iostream>

int main() {
// 创建一个 IPv4 的 endpoint
boost::asio::ip::tcp::endpoint endpoint_v4(boost::asio::ip::address_v4::any(), 8080);

// 创建一个 IPv6 的 endpoint
boost::asio::ip::tcp::endpoint endpoint_v6(boost::asio::ip::address_v6::any(), 8080);

std::cout << "IPv4 Endpoint: " << endpoint_v4.address().to_string() << ":" << endpoint_v4.port() << std::endl;
std::cout << "IPv6 Endpoint: " << endpoint_v6.address().to_string() << ":" << endpoint_v6.port() << std::endl;

return 0;
}

此示例演示了如何创建一个 IPv4 和一个 IPv6 的 tcp::endpoint,并打印它们的 IP 地址和端口号。

注意事项:

  • tcp::endpoint 是一个用于表示网络节点的类,并且常用于创建 acceptorsocket 对象来监听和建立连接。
  • 在实际应用中,可以使用 tcp::endpoint 来标识本地或远程服务器的地址和端口号。

这是一个简单的概述,tcp::endpoint 可以用于标识和操作 TCP 网络上的节点,它是 Boost.Asio 中处理网络连接和数据传输的重要组件之一。

boost::asio::ip::tcp::endpoint 构造函数 输入字符串

boost::asio::ip::tcp::endpoint 类提供了一种从字符串构造端点的方法,它可以解析字符串来创建 TCP 端点。这个构造函数通常用于将用户提供的 IP 地址和端口字符串转换为 tcp::endpoint 对象。

以下是一个示例,演示了如何使用 tcp::endpoint 的构造函数从字符串创建端点:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <boost/asio.hpp>
#include <iostream>

int main() {
// 从字符串创建 TCP 端点
std::string ip_address = "127.0.0.1"; // IP 地址字符串
std::string port_number = "8080"; // 端口号字符串

boost::asio::io_context io_context;

try {
// 使用字符串创建 TCP 端点
boost::asio::ip::tcp::endpoint endpoint(boost::asio::ip::address::from_string(ip_address), std::stoi(port_number));

std::cout << "Created endpoint from strings: " << endpoint.address().to_string() << ":" << endpoint.port() << std::endl;
} catch (std::exception& e) {
std::cerr << "Exception: " << e.what() << std::endl;
}

return 0;
}

在这个示例中,boost::asio::ip::address::from_string() 方法将 IP 地址字符串转换为 address 对象,然后将此对象与端口号一起传递给 tcp::endpoint 的构造函数。最后,使用 endpoint.address().to_string()endpoint.port() 来获取所创建端点的 IP 地址和端口号。

请注意,为了示例简洁,没有进行输入验证。在实际应用程序中,您可能需要添加输入验证来确保字符串包含有效的 IP 地址和端口号。

感谢老板支持!敬礼(^^ゞ