0

0

C# 文件上传的流量控制 C#如何使用令牌桶或漏桶算法来限制上传速率

星降

星降

发布时间:2026-03-02 11:11:42

|

459人浏览过

|

来源于php中文网

原创

throttledstream 是最直接的上传流限速方案,它在每次 read 前检查速率并等待,不阻塞线程,仅拖慢读取节奏;需避免自行实现令牌桶时误用 datetime.now、锁竞争或高频补令牌,且 ratelimitingmiddleware 对上传无效。

c# 文件上传的流量控制 c#如何使用令牌桶或漏桶算法来限制上传速率

ThrottledStream 包裹上传流是最直接的方案

不推荐自己从零实现令牌桶或漏桶——C# 生态已有成熟、轻量、线程安全的封装,比如 Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.ThrottledStream(内部类,不建议直接用),更现实的是采用社区验证过的 ThrottledStream(来自 NuGet 包 System.IO.Streams 或自行实现的简易版)。它的核心逻辑是:每次 Read 前检查是否允许读取指定字节数,若超出速率则等待,不阻塞整个请求,只拖慢流读取节奏。

常见错误现象:自己写 Thread.Sleep 控制间隔,结果在同步 I/O 下卡死整个线程;或在 async/await 中误用 Task.Delay 但没 await,导致限速失效。

  • 使用场景:ASP.NET Core 中处理 IFormFile.OpenReadStream() 返回的流,或 HttpRequest.Body 直接读取大文件时
  • 必须包装在最外层读取点,例如:把原始 Stream 传给 new ThrottledStream(original, bytesPerSecond: 1024 * 1024)
  • 注意:ThrottledStream 通常不实现 Seek,所以不能用于需要随机读取的场景(如 ZIP 解压前校验)
  • 性能影响极小,单次 Read 调用增加约 50–200ns 的判断开销,远低于网络 I/O 本身

RateLimitingMiddleware 对上传无效,别被名字误导

ASP.NET Core 7+ 内置的 RateLimitingMiddleware 只作用于请求频次(requests per second)、并发连接数或 IP 级别,它在管道早期就完成判定,**完全不感知请求体大小或传输过程**。你配置了每秒最多 5 个上传请求,但每个请求仍可能瞬间打满带宽上传 500MB。

典型误用:在 Program.cs 里加了 app.UseRateLimiter() 就以为上传也受控,结果监控看到网卡跑满,而限流中间件日志里毫无异常。

遨虾
遨虾

1688推出的跨境电商AI智能体

下载
  • 该中间件触发点在 HttpContext.Request 头解析完成后、Body 读取前,此时 Body 还没开始接收
  • 它无法与 MultipartReaderFormReader 协同做字节级节流
  • 若真想结合请求级 + 流量级控制,需两层:外层用 RateLimitingMiddleware 控制并发请求数,内层用 ThrottledStream 控制单个请求的上传速率

自实现令牌桶要小心 DateTime.UtcNow 和锁竞争

如果必须手写(例如嵌入非 ASP.NET 环境),令牌桶最简结构只需一个 long _availableTokens、一个 DateTime _lastRefill 和 refill 逻辑。但生产环境容易翻车的点很具体:

  • 别用 DateTime.Now —— 时区和夏令时会导致 _lastRefill 计算错乱,必须用 DateTime.UtcNow
  • 高并发下多个线程同时调用 Consume(long bytes),必须用 Interlocked 操作 _availableTokens,而非 lock —— 否则上传峰值时锁争用会成为瓶颈
  • 令牌补充频率别设太高(如每 1ms 补一次),系统时钟精度有限,高频更新反而导致抖动;推荐每 100ms 补一次,按比例折算令牌数
  • 示例关键片段:
    long now = DateTime.UtcNow.Ticks;
    long elapsedMs = (now - _lastRefill) / TimeSpan.TicksPerMillisecond;
    long tokensToAdd = (long)(elapsedMs * _bytesPerSecond / 1000.0);
    _interlocked.Add(ref _availableTokens, tokensToAdd);
    _lastRefill = now;

前端配合能缓解后端压力,但不能替代服务端限速

前端用 XMLHttpRequest.upload.onprogressfetch + ReadableStream 分块上传,可以切片、暂停、重试,看起来“更可控”。但这只是用户体验优化,**所有分块仍走同一 TCP 连接,服务端不做流控的话,Nginx 或 Kestrel 仍会把数据全收进来再处理**。

真实踩坑:前端限制每秒发 1MB 分片,但服务端未对 Request.Body 做节流,Kestrel 缓冲区积压大量未处理数据,OOM 或超时断连。

  • 必须服务端兜底:哪怕前端完全不可信(如被绕过、脚本篡改),也要保证单请求上传速率可控
  • 若用 Nginx 做反向代理,可配 client_max_body_sizeclient_body_timeout,但它不支持动态速率限制,仅防恶意长连接
  • 真正有效的组合是:前端分片 + 后端 ThrottledStream + 反向代理连接超时兜底

实际部署时,最容易被忽略的是 ThrottledStream 生命周期管理——它必须和上传请求绑定,不能复用或静态持有,否则多个请求会互相干扰令牌计数。另外,HTTPS 加密开销会让实测速率略低于设定值,预留 5–10% 余量更稳妥。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
nginx 重启
nginx 重启

nginx重启对于网站的运维来说是非常重要的,根据不同的需求,可以选择简单重启、平滑重启或定时重启等方式。本专题为大家提供nginx重启的相关的文章、下载、课程内容,供大家免费下载体验。

245

2023.07.27

nginx 配置详解
nginx 配置详解

Nginx的配置是指设置和调整Nginx服务器的行为和功能的过程。通过配置文件,可以定义虚拟主机、HTTP请求处理、反向代理、缓存和负载均衡等功能。Nginx的配置语法简洁而强大,允许管理员根据自己的需要进行灵活的调整。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

519

2023.08.04

nginx配置详解
nginx配置详解

NGINX与其他服务类似,因为它具有以特定格式编写的基于文本的配置文件。本专题为大家提供nginx配置相关的文章,大家可以免费学习。

588

2023.08.04

tomcat和nginx有哪些区别
tomcat和nginx有哪些区别

tomcat和nginx的区别:1、应用领域;2、性能;3、功能;4、配置;5、安全性;6、扩展性;7、部署复杂性;8、社区支持;9、成本;10、日志管理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

244

2024.02.23

nginx报404怎么解决
nginx报404怎么解决

当访问 nginx 网页服务器时遇到 404 错误,表明服务器无法找到请求资源,可以通过以下步骤解决:1. 检查文件是否存在且路径正确;2. 检查文件权限并更改为 644 或 755;3. 检查 nginx 配置,确保根目录设置正确、没有冲突配置等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

629

2024.07.09

Nginx报404错误解决方法
Nginx报404错误解决方法

解决方法:只需要加上这段配置:try_files $uri $uri/ /index.html;即可。想了解更多Nginx的相关内容,可以阅读本专题下面的文章。

3613

2024.08.07

nginx部署php项目教程汇总
nginx部署php项目教程汇总

本专题整合了nginx部署php项目教程汇总,阅读专题下面的文章了解更多详细内容。

53

2026.01.13

nginx配置文件详细教程
nginx配置文件详细教程

本专题整合了nginx配置文件相关教程详细汇总,阅读专题下面的文章了解更多详细内容。

67

2026.01.13

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

48

2026.02.28

热门下载

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

精品课程

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

共94课时 | 10.5万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 20.1万人学习

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

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