[16 使用C++11开发一个简单的通信程序(Proactor模式)] 16.4 C++11结合asio实现一个简单的服务端程序

本文主要是介绍[16 使用C++11开发一个简单的通信程序(Proactor模式)] 16.4 C++11结合asio实现一个简单的服务端程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

需求:服务端监听某个端口,允许多个客户端连接上来,打印客户端发来的数据。

(1)能接收多个客户端。

考虑用一个map来管理socket,每次有新连接时,服务器自动分配一个连接号给这个连接,以方便管理。socket不允许复制,不能直接将socket放到map里,需要外面封装一层。

(2)打印客户端的数据,需要异步读数据。

为简化操作,将socket封装到一个读/写事件处理器中。这时采用同步写,异步读。

读/写事件处理器的实现如下:

#include <array>
#include <functional>
#include <iostream>
using namespace std;#include <boost/asio.hpp>
using namespace boost::asio;
using namespace boost::asio::ip;
using namespace boost;const int MAX_IP_PACK_SIZE = 65536;
const int HEAD_LEN = 4;class RWHandler
{
public:RWHandler(io_service& ios) : m_sock(ios){}~RWHandler(){}void HandleRead(){// 异步读async_read(m_sock, buffer(m_buff), transfer_at_least(HEAD_LEN),[this](const boost::system::error_code& ec, size_t size) {if (ec != nullptr) {HandleError(ec);return;}// 打印客户端发来的数据cout << m_buff.data() + HEAD_LEN << endl;    // 循环发起异步读事件HandleRead();});}void HandleWrite(char* data, int len) {boost::system::error_code ec;// 同步写write(m_sock, buffer(data, len), ec);if (ec != nullptr) {HandleError(ec);}}tcp::socket& GetSocket(){return m_sock;}void CloseSocket(){boost::system::error_code ec;m_sock.shutdown(tcp::socket::shutdown_send, ec);m_sock.close(ec);}void SetConnId(int connId){m_connId = connId;}int GetConnId() const{return m_connId;}void SetCallBackError(std::function<void(int)> f) {m_callbackError = f;}private:void HandlerError(const boost::system::error_code& ec){CloseSocket();cout << ec.message << endl;if (m_callbackError) {m_callbackError(m_connId);}}private:tcp::socket m_sock;//固定长度的读缓冲区std::array<char, MAX_IP_PACK_SIZE> m_buff;int m_connId;std::function<void(int)> m_callbackError;
};

异步操作对象m_socket的生命周期没有处理,socket可能在异步回调返回之前已经释放,这时需要通过shared_from_this来保证对象的生命周期。

有了RWHandler后,服务端接受新连接后的读/写操作就交给RWHandler,服务端Server的实现如下:

#include <boost/asio/buffer.hpp>
#include <unordered_map>
#include <numeric>
#include "Message.hpp"
#include "RWHandler.hpp"const int MaxConnectionNum = 65536;
const int MaxRecvSize = 65536;class Server
{
public:Server(io_service& ios, short port) : m_ios(ios), m_acceptor(ios, tcp::endpoint(tcp::v4(), port), m_cnnIdPool(MaxConnectionNum)){m_cnnIdPool.resize(MaxConnectionNum);// 用顺序递增的值赋值指定范围内的元素std::iota(m_cnnIdPool.begin(), m_cnnIdPool.end(), 1);}~Server(){}void Accept(){cout << "Start Listening..." << endl;std::shared_ptr<RWHandler> handler = CreateHandler();m_acceptor.async_accept(handler->GetSocket(), [this, handler](const boost::system::error_code& error)if (error) {cout << "error: " << error.message() << endl;HandleAcpError(handle, error);return ;}m_handlers.insert(std::make_pair(handler->GetConnId(), handler));cout << "current connect count:" << m_handlers.size() << endl;// 异步读handler->HandleRead();// 等待下一个连接Accept();});}private:void HandlerAcpError(std::shared_ptr<RWHandler> eventHandler, const boost::system::error_code& error){cout << "Error: " << error.message() << endl;// 关闭socket,移除读写事件处理器eventHandler->CloseSocket();StopAccept();}void StopAccept(){boost::system::error_code ec;m_acceptor.cancel(ec);m_acceptor.close(ec);m_ios.stop();}std::shared_ptr<RWHandler> CreateHandler() {int connId = m_cnnIdPool.front();m_cnnIdPool.pop_front();std::shared_ptr<RWHandler> handler = std::make_shared<RWHandler>(m_ios);handler->SetConnId(connId);handler->SetCallBackError([this](int connId){RecyclConnid(connId);});}void RecyclConnid(int connId){auto it = m_handlers.find(connId);if (it != m_handlers.end()) {m_handlers.erase();}cout << "current connect count:" << m_handlers.size() << endl;m_cnnIdPool.push_back(connId);}private:io_service& m_ios;tcp::acceptor m_acceptor;std::unordered_map<int, std::shared_ptr<RWHandler>> m_handlers;list<int> m_cnnIdPool;
};

Server会管理所有连接的客户端。

测试程序如下:

void TestServer()
{io_service ios;Server server(ios, 9900); server.Accept();ios.run();
}

循环发起异步读事件会保证io_service::run一直运行。

这篇关于[16 使用C++11开发一个简单的通信程序(Proactor模式)] 16.4 C++11结合asio实现一个简单的服务端程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1093725

相关文章

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、

C#中lock关键字的使用小结

《C#中lock关键字的使用小结》在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时,其他线程无法访问同一实例的该代码块,下面就来介绍一下lock关键字的使用... 目录使用方式工作原理注意事项示例代码为什么不能lock值类型在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

MySQL 强制使用特定索引的操作

《MySQL强制使用特定索引的操作》MySQL可通过FORCEINDEX、USEINDEX等语法强制查询使用特定索引,但优化器可能不采纳,需结合EXPLAIN分析执行计划,避免性能下降,注意版本差异... 目录1. 使用FORCE INDEX语法2. 使用USE INDEX语法3. 使用IGNORE IND