0

0

C++如何实现带流量整形(Traffic Shaping)功能的网络发送队列?(带宽管理)

尼克

尼克

发布时间:2026-03-09 14:39:11

|

287人浏览过

|

来源于php中文网

原创

c++如何实现带流量整形(traffic shaping)功能的网络发送队列?(带宽管理)

怎么让发送队列按指定带宽发包,而不是一上来就打满

核心是把“发包”从“立刻发”变成“按时间片调度发”。C++ 没有现成的 traffic_shaper 类,得自己在队列和 socket 之间加一层节流逻辑。关键不是压数据,而是控节奏——用定时器驱动出队,而不是靠写 socket 的返回值判断是否阻塞。

常见错误是直接在 send() 前 sleep:这既不准(系统调度延迟大),又浪费线程(尤其高并发时)。正确做法是用异步 I/O 配合时间轮或最小堆管理待发包的计划时间点。

  • std::priority_queue 存待发包,排序依据是「最早允许发送时间戳」
  • 搭配 epoll(Linux)或 kqueue(macOS)监听一个 timerfd,避免忙等
  • 每次 timer 触发后,把所有「当前时间 ≥ 计划发送时间」的包批量出队、调用 send()
  • 如果 send() 返回 EAGAINEWOULDBLOCK,把包重新插回队列,延后重试(别丢弃)

如何计算每个包该安排在什么时间发

流量整形本质是把瞬时突发摊平成恒定速率。假设目标带宽是 10Mbps(即 1.25 MB/s),当前包大小为 1460 字节(典型 TCP MSS),那它至少要和前一个包间隔:1460 / 1250000 ≈ 1.17ms。这个间隔就是「令牌桶」中消耗一个令牌所需时间。

注意单位陷阱:带宽通常给的是 bit/s,但包长是 byte;系统时钟精度也影响下限——clock_gettime(CLOCK_MONOTONIC) 纳秒级可用,但实际调度粒度常是 1–15ms,所以建议最小间隔不低于 2ms,否则调度抖动会明显。

立即学习C++免费学习笔记(深入)”;

Stable Diffusion Online
Stable Diffusion Online

基于Stable Diffusion搭建的AI绘图工具

下载
  • std::chrono::steady_clock 获取当前时间,避免系统时间跳变干扰
  • 别用 usleep()nanosleep() 做逐包延迟,它们不保证唤醒准时
  • 推荐用 timerfd_settime() 设置单次触发的 timerfd,精度远高于用户态 sleep
  • 若用多线程,注意 std::priority_queue 非线程安全,出队/入队需加锁或换用无锁结构(如 moodycamel::ConcurrentQueue

为什么不能只靠 setsockopt(SO_SNDBUF) 控制带宽

SO_SNDBUF 只调内核发送缓冲区大小,它影响的是「能攒多少包等发」,不是「多久发一个」。增大它可能让突发更猛,减小它反而导致频繁 EAGAIN 和重试开销。真正的带宽控制必须发生在应用层决策点——也就是决定「此刻能不能发」的位置。

另一个常见误解是以为 TCP_CONGESTION(如 bbr、cubic)能替代流量整形。它们是端到端拥塞控制,响应的是丢包/延迟,对本地出口带宽无约束力。你在千兆网卡上跑 bbr,照样能打满 1Gbps;而流量整形要求无论网络状况如何,都严格 ≤ 10Mbps。

  • SO_SNDBUF 设太小:频繁触发 send() 失败,逻辑复杂化
  • SO_SNDBUF 设太大:内存占用高,且掩盖了应用层节流缺失的问题
  • 真正可控的入口只有两个:send() 调用时机 + 发送前的数据组织方式

遇到 send() 返回 EAGAIN 后该怎么处理

这不是错误,是正常流控信号。说明内核发送缓冲区满了,但你的队列已经按带宽排好了计划——此时不能丢包,也不能死等,而要把包“退票”,重新算一个稍晚的时间点再排队。

容易踩的坑是直接把包放回队首(导致无限循环尝试),或者用固定延时(比如统一加 10ms),这会让实际速率严重偏离目标。正确做法是基于当前系统时间 + 基础间隔 × 重试次数做指数退避,但上限设为 100ms,避免积压雪崩。

  • 记录包原始入队时间,重试时用「当前时间 + max(基础间隔, 上次失败间隔 × 1.5)」作为新计划时间
  • 给每个包加重试计数,超 5 次就标记为「异常」并走降级路径(如日志告警、切直连 bypass 整形)
  • 务必检查 send() 返回值是否等于 len:TCP 允许部分写入,未写完的部分必须缓存并等待下次发送

最复杂的点其实是时钟同步与调度延迟的耦合——你算出来该在 100.000ms 发,但 timerfd 实际在 100.012ms 触发,这 12μs 在 10Gbps 下就能多发几百字节。真要严控,得在每次触发后动态微调后续包的计划时间,而不是静态排好就不管了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

438

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

764

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

376

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

28

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

103

2026.02.06

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

376

2025.12.24

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号