基于 Boost.Asio 和 Boost.Beast 的异步 HTTP 服务器(学习记录)

news2025/1/8 13:30:09

已完成功能:

        支持 GETPOST 请求的路由与回调处理。

        解析URL请求。

        单例模式 管理核心业务逻辑。

        异步 I/O 技术和 定时器 控制超时。

        通过回调函数注册机制,可以灵活地为不同的 URL 路由注册处理函数。

1. 项目背景

1.1 项目简介

本项目是一个基于 Boost.AsioBoost.Beast 的高性能 HTTP 服务器,实现了对 HTTP 请求的处理、路由功能和回调机制。该服务器使用 单例模式 来管理请求逻辑,并采用 异步 I/O 技术 来提升并发处理能力。支持的 HTTP 请求类型包括 GETPOST,并能够根据 URL 注册不同的处理函数,从而实现灵活的请求路由与回调处理机制。

1.2 项目目标

该项目旨在提供一个 高效可扩展 的 HTTP 服务器,适用于处理大量并发 HTTP 请求,具备以下核心目标:

  • 高并发处理能力:通过使用异步 I/O 和非阻塞 I/O 操作显著提升并发处理能力,避免因同步阻塞导致的性能瓶颈。
  • 可扩展性:支持通过注册回调函数来灵活处理不同的 HTTP 请求,能够根据需要扩展支持更多的 HTTP 请求类型和自定义业务逻辑。
  • 简洁的单例管理:使用单例模式管理全局资源,确保只有一个服务器实例和唯一的请求逻辑处理实例,避免重复初始化资源和管理多个实例的问题。
  • 超时管理:通过 Boost.Asio 提供的定时器机制实现 HTTP 连接的超时控制,确保服务器能够及时关闭不活跃的连接,避免因连接泄漏导致的资源浪费。

2. 项目架构

2.1 架构概述

本项目采用了事件驱动的架构模型,基于 Boost.Asio 提供的异步 I/O 功能实现高效的请求处理。整个系统通过 CServer 启动并接受客户端的连接,HttpConnection 管理每个 HTTP 请求的连接,LogicSystem 处理请求的业务逻辑,利用 单例模式 确保请求处理逻辑的唯一性。各个模块协同工作,通过回调机制对请求进行处理。

服务器架构可以分为以下几个关键模块:

  • CServer:负责启动服务器,监听指定端口并接受客户端连接请求。
  • LogicSystem:使用单例模式管理 HTTP 请求的逻辑处理,负责路由注册与回调函数的执行。
  • HttpConnection:处理每个 HTTP 请求的连接,包括请求解析、响应构建以及连接超时管理。
  • Singleton:一个模板类,提供单例模式的实现,确保系统中逻辑处理系统和其他核心组件只有一个实例。
  • 回调函数:通过回调函数类型 HttpHandler 处理 HTTP 请求,支持 GET 和 POST 请求的灵活处理。

2.2 模块划分

1.网络通信模块

该模块基于 Boost.Asio 库实现,用于管理异步的 TCP 连接和 HTTP 请求的处理。CServer 负责监听端口并接受客户端连接,HttpConnection 负责处理每个连接的 HTTP 请求,采用非阻塞 I/O 操作来保证服务器的高效性能。

关键类:

  • CServer:负责创建 acceptor 并等待连接,管理 tcp::socket
  • HttpConnection:负责处理 HTTP 请求和响应,管理 TCP 连接。
2.HTTP 请求处理模块

该模块基于 Boost.Beast 实现,负责解析 HTTP 请求和构建 HTTP 响应。使用 http::requesthttp::response 类型来处理请求和响应内容。该模块还通过回调函数 HttpHandler 进行路由处理。

关键类:

  • HttpConnection:解析请求并构建响应。
  • LogicSystem:注册并调用处理 GET/POST 请求的回调函数。
3.单例模式模块

通过模板实现的 Singleton 类,确保了系统中 LogicSystem 类等重要组件的唯一性。它通过懒汉式单例模式来延迟初始化,避免了不必要的资源浪费,确保系统的高效运行。

关键类:

  • Singleton:模板类,用于实现 LogicSystem 等核心组件的单例模式。
4.超时管理模块

Boost.Asiosteady_timer 被用于处理连接超时。每个 HTTP 连接都配备了一个定时器,如果连接在规定时间内没有收到有效响应,则会自动关闭连接,防止资源泄漏。

关键类:

  • HttpConnection:内部持有一个 steady_timer,用于处理每个连接的超时。

2.3 数据流

项目的数据流过程如下:

  1. CServer 启动并监听指定端口,等待来自客户端的 TCP 连接。
  2. 客户端连接请求到来时,CServer 使用 acceptor 接受连接,并为每个连接创建一个 HttpConnection 实例。
  3. HttpConnection 启动异步读取操作,接收客户端发送的 HTTP 请求,并将请求数据存储在 beast::flat_buffer 中。
  4. 在接收到完整的 HTTP 请求后,HttpConnection 会解析请求,提取 URL 和请求方法(如 GET 或 POST)。
  5. HttpConnection 将请求转交给 LogicSystemLogicSystem 根据请求的 URL 查找对应的回调函数(HttpHandler),并执行该回调函数进行处理。
  6. 回调函数处理完请求后,HttpConnection 将生成相应的 HTTP 响应,并异步将响应数据发送回客户端。
  7. 如果连接在规定时间内未完成处理,HttpConnection 会触发定时器,主动关闭连接,避免资源浪费。

3. 模块详细设计

由于代码太长,源码我给出链接

源码下载地址,密码:bhmyicon-default.png?t=O83Ahttps://wwta.lanzoue.com/iqugu2k8shij

3.1 CServer 模块

CServer 类是整个 HTTP 服务器的入口,负责启动服务器,监听并接受来自客户端的连接。它依赖于 Boost.Asio 库提供的 tcp::acceptor 来监听指定的端口。当客户端连接到服务器时,CServer 会为每个连接创建一个新的 HttpConnection 对象来处理该连接。

关键函数:
  • Start():启动服务器并开始接受客户端连接。
  • Accept():异步接受客户端连接,当有新的连接请求时调用 HttpConnection 来处理。

CServer 的主要任务是创建和管理连接,每当有新的连接请求时,它会为该连接创建一个 HttpConnection 对象,后者将负责处理 HTTP 请求。

#ifndef CSERVER_H
#define CSERVER_H
#include "const.h"

// CServer 类,负责接受客户端连接并处理连接请求
class CServer :public std::enable_shared_from_this<CServer> {
public:
	CServer(boost::asio::io_context& ioc, unsigned short& port);// 构造函数:初始化服务器并绑定到指定端口
	void Start(); // 启动服务器,开始接受连接
private:
	tcp::acceptor _acceptor;// 用于接受 TCP 连接的接受器
	net::io_context& _ioc;// 提供 I/O 服务的上下文对象
	tcp::socket _socket;// 用于与客户端通信的套接字
};
#endif

3.2 HttpConnection 模块

HttpConnection 类管理每一个客户端的 HTTP 连接。它负责异步读取客户端请求,解析 HTTP 请求,查找 URL 对应的回调函数,并生成 HTTP 响应返回给客户端。此外,HttpConnection 还负责超时管理,通过 Boost.Asiosteady_timer 来确保每个连接的生命周期不会过长,从而避免资源泄漏。

关键函数:
  • Start():启动连接,开始处理 HTTP 请求。
  • HandleReq():处理 HTTP 请求,包括解析请求内容并根据请求方法调用相应的回调函数。
  • CheckDeadline():检查当前连接是否超时,超时则关闭连接。
  • WriteResponse():将生成的 HTTP 响应数据写入客户端。

HttpConnection 的任务是在接收到请求后,解析出请求的方法(GET/POST)、URL 和参数,并将其传递给 LogicSystem 来进行后续的处理。最终,将服务器的响应发送给客户端。

#ifndef HTTPCONNECTION_H
#define HTTPCONNECTION_H
#include "const.h"

// HttpConnection 类,负责与客户端的 HTTP 连接
// 提供处理 HTTP 请求、定时器管理等功能
class HttpConnection :public std::enable_shared_from_this<HttpConnection> {
	friend class LogicSystem;// 允许 LogicSystem 访问 HttpConnection 的私有成员
public:
	HttpConnection(tcp::socket socket);// 构造函数:接受 TCP 套接字,初始化连接
	void Start(); // 启动连接处理
private:
	void CheckDeadline();// 检查是否超时,定时器功能
	void WriteResponse();// 写入响应数据
	void HandleReq(); // 处理 HTTP 请求
	void PreParseGetParam();//处理参数解析
	tcp::socket _socket;// 客户端套接字
	beast::flat_buffer _buffer{ 8192 };// 接收数据的缓冲区,最大缓存 8192 字节
	http::request<http::dynamic_body> _request;// HTTP 请求对象,存储接收到的 HTTP 请求数据
	http::response<http::dynamic_body> _response;// HTTP 响应对象,存储响应数据并发送给客户端
	// 定时器,检查连接是否超时,60 秒后自动关闭连接
	net::steady_timer deadline_{
		_socket.get_executor(),std::chrono::seconds(60)
	};
	std::string _get_url; // 存储请求的 URL
	std::unordered_map<std::string, std::string> _get_params;// 存储 GET 请求的查询参数,以键值对形式存储
};

#endif

3.3 LogicSystem 模块

LogicSystem 类是整个系统的核心,负责管理 HTTP 请求的路由与回调函数。它采用 单例模式 确保系统中只有一个逻辑处理实例,并通过 std::map 存储 GET 和 POST 请求的回调函数。每当 HTTP 请求到来时,HttpConnection 会将请求的 URL 传递给 LogicSystem,然后根据请求方法(GET/POST)查找并执行对应的回调函数。

关键函数:
  • RegGet():注册 GET 请求的回调函数。
  • RegPost():注册 POST 请求的回调函数。
  • HandleGet():处理 GET 请求,执行对应的回调函数。
  • HandlePost():处理 POST 请求,执行对应的回调函数。

LogicSystem 是一个单例类,它为每个请求类型(GET/POST)提供了 URL 到回调函数的映射表。当请求到来时,LogicSystem 会根据请求的 URL 查找相应的回调函数并执行,从而实现不同 URL 对应不同的处理逻辑。

#ifndef LOGICSYSTEM_H
#define LOGICSTSTEM_H
#include "Singleton.h"
#include <functional>
#include <map>
#include "const.h"

class HttpConnection;

typedef std::function<void(std::shared_ptr<HttpConnection>)> HttpHandler;// 定义一个回调函数类型,处理 HTTP 请求
// HttpHandler 通过 shared_ptr 传递 HttpConnection 对象

// LogicSystem 类,用于管理 HTTP 请求的逻辑处理
// 采用单例模式,确保系统只有一个实例
class LogicSystem : public Singleton<LogicSystem> {
	friend class Singleton<LogicSystem>; // 允许 Singleton 类访问 LogicSystem 的构造函数
public:
	~LogicSystem();// 析构函数,负责释放资源
	bool HandleGet(std::string path, std::shared_ptr<HttpConnection> con); // 处理GET请求的函数
	bool HandlePost(std::string path, std::shared_ptr<HttpConnection> con);//处理POST请求的函数
	void RegGet(std::string url, HttpHandler handler);// 注册 GET 请求的回调函数
	void RegPost(std::string url, HttpHandler handler);// 注册POST请求的回调函数
private:
	LogicSystem(); // 构造函数,私有化以防外部直接创建实例
	//key为路由,value为回调函数
	std::map<std::string, HttpHandler> _post_handlers; // 存储 POST 请求的处理函数
	std::map<std::string, HttpHandler> _get_handlers; // 存储 GET 请求的处理函数
};
#endif

3.4 Singleton 模块

Singleton 模块通过模板实现了单例模式,确保系统中的核心组件(如 LogicSystem)只有一个实例。该模板类使用懒汉式初始化,确保在首次访问时才创建实例,并且每次访问都返回同一个实例。

关键函数:
  • GetInstance():获取单例实例,如果实例尚未创建,则进行初始化。
  • PrintAddress():打印当前单例实例的地址,便于调试。

Singleton 模块保证了系统中只有一个实例,可以有效避免重复实例化带来的资源浪费。通过使用 std::shared_ptr 管理实例的生命周期,确保实例的销毁是在所有引用都释放后进行的。

#ifndef SINGLETON_H
#define SINGLETON_H
#include <mutex>
#include <iostream>
#include <memory>

// Singleton 模板类,确保一个类只有一个实例
// 使用 std::shared_ptr 来管理实例生命周期
template <typename T>
class Singleton {
protected:
	Singleton() = default;  // 默认构造函数,防止外部实例化
	Singleton(const Singleton<T>&) = delete; // 禁止拷贝构造
	Singleton& operator=(const Singleton<T>& st) = delete; // 禁止拷贝赋值

	static std::shared_ptr<T> _instance;// 存储单例实例的静态变量
public:
	// 获取单例实例,线程安全的懒汉式初始化
	static std::shared_ptr<T> GetInstance() {
		static std::once_flag s_flag;// 用于确保单例实例只被创建一次
		std::call_once(s_flag, [&]() {
			// 创建单例实例并保存在 _instance 中
			_instance = std::shared_ptr<T>(new T);
			});
		return _instance;// 返回单例实例
	}
	// 打印当前单例实例的地址
	void PrintAddress() {
		std::cout << _instance.get() << std::endl;
	}
	// 析构函数,打印销毁信息
	~Singleton() {
		std::cout << "this is singleton destruct" << std::endl;
	}
};

// 为单例提供静态变量的初始化
template<typename T>
std::shared_ptr<T>Singleton<T>::_instance = nullptr;
#endif

4.测试

输入http://localhost:8080/get_test

显示receive get_test req

输入http://localhost:8080/get_test?key1=value1&key2=value2

显示

receive get_test req
param1key iskey1, value isvalue1
param2key iskey2, value isvalue2 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2273236.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux标准IOday1

1:思维导图 2:将 student.c这个练习题&#xff0c;改成链表后实现 头文件link.h #ifndef __STRUCT_H__ #define __STRUCT_H__ #include <stdio.h> #include <stdlib.h> typedef struct Student{char name[20];double math;double chinese;double english;double…

全局变量(PHP)(小迪网络安全笔记~

免责声明&#xff1a;本文章仅用于交流学习&#xff0c;因文章内容而产生的任何违法&未授权行为&#xff0c;与文章作者无关&#xff01;&#xff01;&#xff01; 附&#xff1a;完整笔记目录~ ps&#xff1a;本人小白&#xff0c;笔记均在个人理解基础上整理&#xff0c;…

gateway的路径匹配介绍

gateway是一个单独服务。通过网关端口和predicates进行匹配服务 1先看配置。看我注解你就明白了。其实就是/order/**配置机制直接匹配到orderservice服务。 2我试着请求一个路径&#xff0c;请求成功。下面第三步是请求的接口。 3接口。

RabbitMQ-基本使用

RabbitMQ: One broker to queue them all | RabbitMQ 官方 安装到Docker中 docker run \-e RABBITMQ_DEFAULT_USERrabbit \-e RABBITMQ_DEFAULT_PASSrabbit \-v mq-plugins:/plugins \--name mq \--hostname mq \-p 15672:15672 \-p 5672:5672 \--network mynet\-d \rabbitmq:3…

模式识别-Ch2-分类错误率

分类错误率 最小错误率贝叶斯决策 样本 x x x的错误率&#xff1a; 任一决策都可能会有错误。 P ( error ∣ x ) { P ( w 2 ∣ x ) , if we decide x as w 1 P ( w 1 ∣ x ) , if we decide x as w 2 P(\text{error}|\mathbf{x})\begin{cases} P(w_2|\mathbf{x}), &…

CAD批量打印可检索的PDF文件

本文虽介绍CAD使用方法&#xff0c;但还是劝告大家尽早放弃使用CAD软件。。。。太TM难用了 当你打开CAD时发现如下一堆图纸&#xff0c;但是不想一个一个打印时。你可以按照下面操作实现自动识别图框实现批量打印。 1.安装批量打印插件 2.安装后打开CAD&#xff0c;输入命令Bp…

BERT:深度双向Transformer的预训练用于语言理解

摘要 我们介绍了一种新的语言表示模型&#xff0c;名为BERT&#xff0c;全称为来自Transformer的双向编码器表示。与最近的语言表示模型&#xff08;Peters等&#xff0c;2018a&#xff1b;Radford等&#xff0c;2018&#xff09;不同&#xff0c;BERT旨在通过在所有层中联合调…

搭建企业AI助理的创新应用与案例分析

在大健康零售行业&#xff0c;企业面临着日益增长的市场需求和复杂的供应链管理挑战。AI助理的应用不仅能够提升客户服务效率&#xff0c;还能优化供应链管理&#xff0c;降低运营成本。 一、AI助理在大健康零售行业的创新应用 个性化健康咨询 AI助理可以通过分析客户的健康…

apex安装

安装过程复杂曲折&#xff0c;网上说的很多办法&#xff0c;貌似成功了&#xff0c;实际还是没起作用。 先说成功过程&#xff0c;执行下面命令&#xff0c;安装成功&#xff08;当然&#xff0c;前提是你要先配置好编译环境&#xff09;&#xff1a; &#xff08;我的环境&a…

select下拉框,首次进入页面没有显示value的情况

bug场景&#xff1a; 类似这种bug情况排查如下&#xff1a; 首先 理解含义 options就是存放键值对的&#xff0c;id就是key&#xff0c;对上了它就自动把label显示 而且如果你用来当作key和label的字段&#xff0c;与后端返回的不一致&#xff0c;还可以进行更改 其次 排查接…

krpano 实现文字热点中的三角形和竖杆

krpano 实现文字热点中的三角形和竖杆 实现文字热点中的三角形和竖杆 一个后端写前端真的是脑阔疼 一个后端写前端真的是脑阔疼 一个后端写前端真的是脑阔疼 实现文字热点中的三角形和竖杆 上图看效果 v&#xff1a;2549789059

playwright 录制

一、新建项目TestProject3 二、准备swagger 三、开始录制 打开PowerShell 7 (x64) cd D:\xxx\xxx\VS2022Projects\TestProject3\TestProject3\bin\Debug\net8.0 pwsh playwright.ps1 codegen --targetcsharp -b chromium localhost:5252/swagger/index.html #支持的语言 java…

uni-app 资源引用(绝对路径和相对路径)方法汇总

文章目录 一、前言&#x1f343;二、绝对路径和相对路径2.1 绝对路径2.2 相对路径 三、引用组件四、引用js4.1 js 文件引入4.2 NPM支持 五、引用css六、引用json6.1 json文件引入 七、引用静态资源7.1 模板内引入静态资源7.2 css 引入静态资源7.3 js/uts 引入静态资源7.4 静态资…

go如何从入门进阶到高级

针对Go语言的学习&#xff0c;不同阶段应采取不同的学习方式&#xff0c;以达到最佳效果.本文将Go的学习分为入门、实战、进阶三个阶段&#xff0c;下面分别详细介绍 一、社区 Go语言中文网 作为专注于Go语言学习与推广的平台&#xff0c;Go语言中文网为开发者提供了丰富的中…

现代密码学期末重点(备考ing)

现代密码学期末重点&#xff0c;个人备考笔记哦 密码学概念四种密码学攻击方法什么是公钥密码&#xff1f;什么是对称密码&#xff1f;什么是无条件密码&#xff1f; 中国剩余定理&#xff08;必考&#xff09;什么是原根什么是阶 经典密码学密码体制什么是列置换&#xff1f; …

基于SMT32U575RIT单片机-中断练习

任务 查看手册对所有的拓展板上和相对应的底板的引脚对应的端口找到以下结论 通过STM32MX软件对各个引脚进行相应的配置 1.第一种切换模式电脑发送 #include "main.h" #include "icache.h" #include "usart.h" #include "gpio.h"/*…

KCP解读:拥塞控制

本文是系列文章中的一篇&#xff0c;内容由浅到深进行剖析&#xff0c;为了方便理解建议按顺序一一阅读。 KCP技术原理 KCP解读&#xff1a;基础消息收发 KCP解读&#xff1a;重传机制 KCP解读&#xff1a;滑动窗口 KCP解读&#xff1a;拥塞控制 本文摘取https://xiaolincodin…

HCIA-Access V2.5_8_2_EPON基本架构和关键参数

EPON数据利用方式 EPON和GPON同样只有一根光纤&#xff0c;所以为了避免双向发送数据出现冲突&#xff0c;我们同样采用WDM技术&#xff0c;那么主要利用两个波长&#xff0c;一个是1490纳米的波长&#xff0c;一个是1310纳米的波长&#xff0c;下行OLT给ONU发送数据的时候&…

如何快速上手一个鸿蒙工程

作为一名鸿蒙程序猿&#xff0c;当你换了一家公司&#xff0c;或者被交接了一个已有的业务。前辈在找你之前十分钟写了一个他都看不懂的交接文档&#xff0c;然后把一个鸿蒙工程交接给你了&#xff0c;说以后就是你负责了。之后几天你的状态大概就是下边这样的&#xff0c;一堆…

FPGA实现UART对应的电路和单片机内部配合寄存器实现的电路到底有何区别?

一、UART相关介绍 UART是我们常用的全双工异步串行总线&#xff0c;常用TTL电平标准&#xff0c;由TXD和RXD两根收发数据线组成。 那么&#xff0c;利用硬件描述语言实现UART对应的电路和51单片机内部配合寄存器实现的电路到底有何区别呢&#xff1f;接下来我们对照看一下。 …