当前位置: 首页 > news >正文

CPP网络编程-异步sever

代码:

// AsyncServer.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//#include <iostream>
#include<boost/asio.hpp>
#include"Session.h"int main()
{try {boost::asio::io_context ioc;using namespace std;Server s(ioc, 10086);ioc.run();}catch(std::exception& e) {std::cout << e.what() << std::endl;}
}
//Session.h
#pragma once
#include<iostream>
#include<boost/asio.hpp>using boost::asio::ip::tcp;
class Session
{
public:Session(boost::asio::io_context& ioc):_socket(ioc) {}tcp::socket& Socket() {return _socket;}void Start();private:void handle_read(const boost::system::error_code& error, size_t bytes_transferred);void handle_write(const boost::system::error_code& error);//异步发送,发送完tcp::socket _socket;enum { max_length = 1024 };char _data[max_length];
};//建立连接
class Server {
public:Server(boost::asio::io_context& ioc, short port);
private:void start_accept();void handle_accept(Session* new_session, const boost::system::error_code& error);boost::asio::io_context& _ioc;tcp::acceptor _acceptor;
};
//Session.cpp
#include "Session.h"
#include<iostream>
using namespace std;void Session::Start() {memset(_data, 0, max_length);//置0初始化_socket.async_read_some(boost::asio::buffer(_data, max_length),std::bind(&Session::handle_read, this,placeholders::_1, placeholders::_2));//_socket——>异步读,读到的数据放在_data中,当读完后,调用handle_read
}void Session::handle_read(const boost::system::error_code& error,size_t bytes_transferred ){if (!error) {cout << "server receive data is " << _data << endl;boost::asio::async_write(_socket, boost::asio::buffer(_data, bytes_transferred),std::bind(&Session::handle_write, this, placeholders::_1));}else {cout<<"read error"<<endl;delete this;}
}void Session::handle_write(const boost::system::error_code& error) {if (!error) {memset(_data,0, max_length);_socket.async_read_some(boost::asio::buffer(_data, max_length),std::bind(&Session::handle_read, this, placeholders::_1, placeholders::_2));}else {cout << "write error" << error.value()<<endl;delete this;}
}Server::Server(boost::asio::io_context& ioc, short port):_ioc(ioc),_acceptor(ioc,tcp::endpoint(tcp::v4(),port)) {cout << "server start,port:" <<port<< endl;start_accept();
}
void Server::start_accept() {/*先创建一个new_session,然后调用async_accept,当有新的连接时,会调用handle_accept,*/Session* new_session = new Session(_ioc);_acceptor.async_accept(new_session->Socket(),std::bind(&Server::handle_accept,this,new_session,placeholders::_1));
}
void Server::handle_accept(Session* new_session, const boost::system::error_code& error) {if (!error) {new_session->Start();}else {delete new_session;}start_accept();
}

Session.h

Session 类

成员

作用
_socket与客户端通信的 TCP 套接字
_data缓冲区,最多存 1024 字节数据
Socket()返回 socket 引用,供 acceptor 使用
Start()启动异步读操作
handle_read读完成后的回调函数
handle_write写完成后的回调函数

Server 类

成员作用
_ioc引用外部 io_context
_acceptor监听指定端口的 TCP 接收器
start_accept()主动发起异步接受连接请求
handle_accept()接受连接完成后的回调

Session.cpp

void Session::Start() {memset(_data, 0, max_length);_socket.async_read_some(boost::asio::buffer(_data, max_length),std::bind(&Session::handle_read, this, placeholders::_1, placeholders::_2));
}

启动对当前socket的异步读取,等待客户端发送数据
async_read_some非阻塞读,只要有数据就触发回调,_data缓冲区,然后绑定回调函数handle_read

此函数只调用一次(在连接建立后),之后通过 handle_read → handle_write → 再次 async_read_some 形成循环。

void Session::handle_read(const boost::system::error_code& error, size_t bytes_transferred) {if (!error) {cout << "server receive data is " << _data << endl;boost::asio::async_write(_socket, boost::asio::buffer(_data, bytes_transferred),std::bind(&Session::handle_write, this, placeholders::_1));} else {cout << "read error" << endl;delete this;}
}

如果读取成功打印数据,使用async_write将数据完整回传给客户端,发送完调用handle_write

void Session::handle_write(const boost::system::error_code& error) {if (!error) {memset(_data, 0, max_length);_socket.async_read_some(boost::asio::buffer(_data, max_length),std::bind(&Session::handle_read, this, placeholders::_1, placeholders::_2));} else {cout << "write error" << error.value() << endl;delete this;}
}
  • 写操作完成后的回调。
  • 若成功:清空缓冲区,重新开始 async_read_some,等待下一条消息
Server::Server(boost::asio::io_context& ioc, short port): _ioc(ioc), _acceptor(ioc, tcp::endpoint(tcp::v4(), port)) {cout << "server start,port:" << port << endl;start_accept();
}

初始化 _acceptor,绑定到 IPv4

void Server::start_accept() {Session* new_session = new Session(_ioc);_acceptor.async_accept(new_session->Socket(),std::bind(&Server::handle_accept, this, new_session, placeholders::_1));
}
  • 每次调用都 new 一个 Session
  • 将 Session 的 _socket 作为参数传给 async_accept
  • 设置回调为 handle_accept,并把 new_session 指针传进去
void Server::handle_accept(Session* new_session, const boost::system::error_code& error) {if (!error) {new_session->Start();  // 启动会话} else {delete new_session;    // 接受失败,清理}start_accept();  // 无论成功与否,继续监听下一个连接
}
  • 如果接受连接成功 → 启动 Session::Start() 开始读取;
  • 如果失败(如监听关闭)→ 删除刚创建的 Session
                     +------------------+|   Server启动     |+--------+---------+|v创建_acceptor,绑定端口10086|v调用 start_accept()|vnew Session(_ioc)     ← 堆上创建|v_acceptor.async_accept(socket, handle_accept)|| (等待客户端连接)↓[客户端连接到来]|v触发 handle_accept()|+-------+--------+| 是否出错?     |+-------+--------+|是 /        \ 否v          vdelete session   new_session->Start()|v_socket.async_read_some(...)|| (等待客户端发数据)↓[客户端发送数据]|v触发 handle_read(...)|v打印数据 → async_write(...)|| (等待发送完成)↓触发 handle_write(...)|v清空缓冲 → 再次 async_read_some(...)|(循环往复)

http://www.dtcms.com/a/317665.html

相关文章:

  • FPGA学习笔记——VGA彩条显示
  • python:非常流行和重要的Python机器学习库scikit-learn 介绍
  • STM32学习笔记3-GPIO输入部分
  • WMS及UI渲染底层原理学习
  • 【STM32 LWIP配置】STM32H723ZG + Ethernet +LWIP 配置 cubemx
  • 无人机图传的得力助手:5G 便携式多卡高清视频融合终端的协同应用
  • 中宇联5G云宽带+4G路由器:解锁企业办公高效协同与门店体验升级
  • 图解 Claude Code 子智能体 Sub-agent
  • [ java GUI ] 图形用户界面
  • 【软考系统架构设计师备考笔记4】 - 英语语法一篇通
  • ctfshow_vip题目限免-----SVN漏洞,git泄露
  • Git Cherry-Pick 指南
  • Leetcode——菜鸟笔记1
  • Git 分支管理:从新开发分支迁移为主分支的完整指南
  • 鸿蒙app 开发中 全局弹窗类的封装 基于PromptAction
  • C#之基础语法
  • 机器学习之朴素贝叶斯
  • Suno API V5模型 php源码 —— 使用灵感模式进行出创作
  • 基于PHP的论坛社交网站系统开发与设计
  • 排序算法详解
  • 媒体资产管理系统和OCR文字识别的结合
  • Ethereum: L1 与 L2 的安全纽带, Rollups 技术下的协作与区别全解析
  • 解决启动docker报错Cannot connect to the Docker daemon问题
  • 阿里 Qwen-Image:开源 20B 模型引领图像生成新纪元,中文渲染超越 GPT-4o!
  • 数据结构与算法的认识
  • 手动开发一个TCP服务器调试工具(二):无界面 TCP 通信服最小实现
  • ETF期权分仓的风险如何管理?
  • 基于Hadoop的股票大数据分析可视化及多模型的股票预测研究与实现
  • 四十、【高级特性篇】接口用例数据驱动:引入随机变量与动态数据生成
  • 生成式模型 ?判别式模型?用【猫狗分类器】帮助理解!