您的位置:首页 >C# Span T用法及性能优化技巧
发布于2026-02-05 阅读(0)
扫一扫,手机访问
<p>Span是C# 7.2引入的栈分配、零分配、类型安全的内存切片类型,本质是“指向连续内存块的轻量视图”,适用于短期、局部、高性能场景如字符串解析和二进制协议解包。</p>

Span
常见误用:拿 Span 替代 List 或长期缓存数据——这会编译失败或运行时报 System.ArgumentException: Span must be stack-only。
创建方式直接决定性能和安全性,关键看底层是否支持栈视图:
Span 可安全来自:栈数组(stackalloc byte[256])、堆数组(new byte[1024].AsSpan())、Memory(调用 .Span)string → Span:用 "hello".AsSpan(),零拷贝;但注意 Span 是只读视图,不能修改字符串内容ReadOnlySpan 更常用:避免意外写入,且兼容更多 API(如 int.TryParse(ReadOnlySpan, out int) )StringBuilder.ToString() 直接 .AsSpan()——返回的是新字符串,生命周期短,且可能触发 GC;应改用 StringBuilder 的 Span 接口(如 builder.AsSpan(),需 .NET 6+)看似高效,但一不留神就掉回堆分配或引发异常:
Span 传给接受 object 或泛型约束为 class 的方法 → 编译失败(error CS8345: Field or property cannot be of type 'Span' unless it is declared as 'ref readonly' )async 方法里捕获 Span 并 await 后使用 → 运行时报 System.InvalidOperationException: Cannot use a Span across await span.Slice(i, len) 不产生新内存,但每次生成新结构体(值类型),若在 tight loop 中大量创建,可能增加栈压力(尤其嵌套深时)ToArray():span.ToArray() 触发堆分配 → 应优先用 Memory + MemoryPool.Shared.Rent() 配合 Span 操作,最后再拷贝比如解析形如 "key=value&flag=true" 的 query string,不用 string.Split('&') 和 Substring(每步都分配新字符串):
static void ParseQuery(ReadOnlySpaninput) { int start = 0; while (input.Slice(start).IndexOf('&') is int ampIdx && ampIdx >= 0) { var pair = input.Slice(start, ampIdx); int eqIdx = pair.IndexOf('='); if (eqIdx > 0) { ReadOnlySpan key = pair.Slice(0, eqIdx).Trim(); ReadOnlySpan value = pair.Slice(eqIdx + 1).Trim(); // 直接用 ReadOnlySpan 调用 int.TryParse / bool.TryParse 等 } start += ampIdx + 1; } }
这个版本全程无字符串分配,GC 压力趋近于零。但要注意:所有参与解析的 API 必须原生支持 ReadOnlySpan(如 .NET Core 2.1+ 的 int.TryParse);旧版框架需升级或手动实现字符遍历逻辑。
真正难的不是写对 Span,而是判断哪一段逻辑值得重构——往往要先用 dotTrace 或 dotMemory 抓到热点分配点,再针对性替换。盲目全量改成 Span 可能让代码更难懂,却换不来实际收益。
上一篇:大智慧数据加载慢怎么解决
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9