0

0

c# C# 高并发下的加密解密操作性能瓶颈

星降

星降

发布时间:2026-01-14 10:12:09

|

542人浏览过

|

来源于php中文网

原创

RijndaelManaged 高并发下因同步锁和频繁实例化成瓶颈;AesGcm 替代时需避免同步 nonce 生成并严格处理 tag;MemoryPool 仅在固定长度高频场景有效,关键在 nonce 管理与 buffer 生命周期控制。

c# c# 高并发下的加密解密操作性能瓶颈

为什么 RijndaelManaged 在高并发下会成为瓶颈

它不是线程安全的,每次调用 CreateEncryptor()CreateDecryptor() 都会内部新建状态对象,且其默认实现使用同步锁保护静态资源(如填充模式、密钥调度缓存)。在千级 QPS 场景下,大量短生命周期实例频繁争抢锁,Monitor.Enter 开销会明显抬高 CPU 和延迟。

  • 不要复用同一个 RijndaelManaged 实例跨线程调用 —— 会抛 CryptographicException: The object is not in the correct state
  • 不要在循环中反复 new RijndaelManaged() 并手动 Dispose() —— GC 压力大,且构造成本固定约 0.1–0.3ms
  • .NET Core 3.0+ 中已标记为过时,RijndaelManaged 不再推荐用于新项目

AesGcm 替代时要注意哪些参数陷阱

AesGcm硬件加速友好的 AEAD 算法,但它的 API 设计对并发不友好:必须为每次加密生成唯一 nonce(通常 12 字节),且不能重复;解密时需原样传入同一 nonce + 认证标签(16 字节)。若用 RandomNumberGenerator 同步生成 nonce,极易成为新瓶颈。

  • 避免在加密逻辑里调用 RandomNumberGenerator.GetBytes(nonce) —— 它是同步锁保护的全局 RNG
  • 改用预分配 Span + RandomNumberGenerator.Fill()(.NET 6+)或分片 counter-based nonce(如每线程维护递增 long,转为 big-endian 12 字节)
  • AesGcm.Encrypt() 输出包含密文 + tag,务必把 tag 追加到结果末尾;解密前必须严格切分最后 16 字节作为 tag,否则抛 CryptographicException: Authentication failed

MemoryPool 能否真正降低 GC 压力

能,但仅当你的加密数据长度相对固定(如统一 256–2048 字节)且 QPS > 500 时才值得引入。如果每次加密输出长度波动大,或者用 ArrayPool.Shared 未控制租借大小,反而可能因碎片导致内存浪费或 fallback 到 new byte[]。

  • 优先用 MemoryPool.Shared.Rent(int minBufferSize),而非 ArrayPool.Shared.Rent() —— 前者支持超大 buffer(>85KB),后者只管小对象堆
  • 解密后立即 pool.Return(buffer, clearArray: true),避免敏感数据残留内存
  • 别在 using var output = pool.Rent(...) 外部捕获 output.Memory 引用 —— Return 后该内存可能被复用,引发脏读
var pool = MemoryPool.Shared;
var input = Encoding.UTF8.GetBytes("secret data");
var nonce = stackalloc byte[12];
// 使用线程本地 counter 构造 nonce,非 Random
Unsafe.WriteUnaligned(ref nonce[4], ThreadLocalCounter.Next());

var outputBuffer = pool.Rent(input.Length + 16); // 密文 + tag try { var gcm = AesGcm.Create(); gcm.Encrypt( nonce, input, outputBuffer.Memory.Span[..input.Length], outputBuffer.Memory.Span[input.Length..], associatedData: default); } finally { pool.Return(outputBuffer, clearArray: true); }

高并发加解密真正的复杂点不在算法选型,而在于 nonce 管理方式buffer 生命周期边界。这两个地方一旦和线程模型耦合错位,轻则性能抖动,重则数据损坏或内存泄露。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

52

2025.08.29

C++中int的含义
C++中int的含义

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

194

2025.08.29

堆和栈的区别
堆和栈的区别

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

387

2023.07.18

堆和栈区别
堆和栈区别

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

571

2023.08.10

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

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

480

2023.08.10

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

400

2023.08.14

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

2

2026.01.14

热门下载

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

精品课程

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

共18课时 | 4.5万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

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

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