Boost系列之asio库简易使用及遇到的问题(UDP)和思考

Boost系列之asio库简易使用及遇到的问题(UDP)

1、个人学习asio的一些步骤和看法

首先我都是直接文档怼上:Boost.Asio - 1.68.0

在看文档需要了解一下大概;

Overview - 1.68.0 (boost.org)Using Boost.Asio - 1.68.0理论部分居多,简单阅读可以了解asio的内部细节;

Tutorial - 1.68.0 (boost.org)Examples - 1.68.0 (boost.org)这两个章节主要讲解如何使用asio库;

Reference - 1.68.0 (boost.org)这个章节对我来说在开发应用打开最多的章节,里面包含了asio库API的方方面面,值得粗读和细读;

剩余章节没有细究;

2、我用到了哪些asio特性

  • io_context
  • buffer(异步读写)
  • ip::udp ip::endpoint socket属性操作
  • 以及部分timers内容

3、下面是使用asio作为udp-server的sample 片段代码

c++ 复制代码
void gvstream::CaptureMultiPartFrame() {
	m_receiver_socket.async_receive_from(
		boost::asio::buffer(m_message.data(), m_message.size()),
		m_tramitter_endpoint,
        // 由HandleMultipartMessage函数处理接收到的信息
		boost::bind(&aingvstream::HandleMultipartMessage, this,
			boost::asio::placeholders::error,
			boost::asio::placeholders::bytes_transferred)
	);
}

void gvstream::HandleMultipartMessage(const boost::system::error_code& err, size_t bytesTransferred) {
	char* multiPartPacket = new char[m_packet_size];
	memcpy(multiPartPacket, m_message.data(), bytesTransferred);
    // 详细解析收到的信息包
	uint64_t blockId = ProcessPacket(multiPartPacket, bytesTransferred);
    // 检测一个帧数据被拆分成多个packet后,server接收到packet组成帧的完整性
	CheckFrameCompletion(blockId);
    // 再次异步接收
	CaptureMultiPartFrame();
}

上面的片段是Asio-UDP-Server最精彩的桥段,使用异步接收的方式处理收到的消息,再通过回调函数再次异步接收。

4、问题和思考

问题:在代码片段注释里已经明确的说明了代码的意图。这时我们可以想到所有的异步接收实际上是同步处理(在HandleMultipartMessage 里调用CaptureMultiPartFrame ),那么对于UDP这种没有不稳定,不保证数据重传、正确性的协议,丢包时有发生;而且在上面的片段代码中,当client大量数据传输到server端时,丢包率极高。这是因为ProcessPacket 占据了大量的CPU时间,导致在新包到来时还没有调用CaptureMultiPartFrame从而导致丢包。

思考:这里的简单应用问题很多,往往需要自定义实现重传或直接TCP协议。再不然就是使用线程池+异步接收的方式实现降低丢包率甚至不丢包。这里都非常的繁琐,这是在windows层面我们可以使用系统自带的IOCP实现UDP-Server基本可以完美的解决问题,它内部使用线程池+消息队列的方式防止大量数据到来来不及处理的问题。

要想了解IOCP-UDP可以参考我的文章:https://blog.csdn.net/youzai2017/article/details/138321593

相关推荐
编程之路,妙趣横生16 小时前
STL(五) priority_queue 基本用法 + 模拟实现
c++
一念一花一世界16 小时前
Arbess从初级到进阶(9) - 使用Arbess+GitLab实现C++项目自动化部署
c++·ci/cd·gitlab·arbess
大锦终17 小时前
【Linux】Reactor
linux·运维·服务器·c++
沐怡旸17 小时前
【穿越Effective C++】23.宁以non-member、non-friend替换member函数
c++·面试
至天17 小时前
Windows 10/11 终端(Windows Terminal)右键菜单缺失恢复方法
windows·终端·windows 11·windows 10·terminal·右键菜单·windowsterminal
青小俊18 小时前
【代码随想录c++刷题】-二分查找 移除元素 有序数组的平方 - 第一章 数组 part 01
c++·算法·leetcode
阿珊和她的猫18 小时前
WebSocket 与轮询:实时通信技术的对比与选择
网络·websocket·网络协议
光年像素18 小时前
WMIC(Windows Management Instrumentation Command-line)命令大全
windows
赖small强18 小时前
【Linux C/C++开发】第16章:多线程编程基础
linux·c语言·c++·多线程编程·进程和线程的本质区别
AA陈超19 小时前
以 Lyra 的架构为基础,创建一个名为 “Aura“ 的英雄并实现发射火球技能
c++·笔记·学习·ue5·lyra