.NET高性能内存管理

.NET高性能内存管理

你是不是还不了解spanMemeoryArrayPool这些用法,随着.net的不断升级,像这些提高性能的用法也层出不穷,本文重点介绍一些能够提升.net性能的用法

开篇之前,先说下提高性能的宗旨,无非就是尽量减少堆上重新分配与数据复制,让内存得以重用或在栈上分配,从而降低 GC 压力、提升访问速度

Span、ReadOnlySpan

它是栈上的结构(ref struct),所以不会触发GC,可以提供数组、字符串等的零拷贝切片,生命周期仅仅局限于当前的栈,也就是当前方法结束,就会清除,不可以跨方法或者异步操作传递。
ReadOnlySpan<T>ref struct,只能用于:局部变量、方法参数/返回值,或 ref struct 的实例字段。

整体比较好用,不过我觉得比较鸡肋的一点是只能在同步方法中使用,不能再异步方法中传递,因为await本质会把方法状态机移到堆上,不过又推出了Memory解决了这个问题,见下文。

csharp 复制代码
{
    string csv = "1,2,3,4,5";
    ReadOnlySpan<char> span = csv.AsSpan();
    ReadOnlySpan<char> firstTwo = span.Slice(0, 3);
    firstTwo.ToString().Display(); // 输出 "1,2"此时没有有新的字符串分配
    csv.Substring(0,3).Display(); // 输出 "1,2",此时有新的字符串分配
}

Span<T>仅创建视图,不分配新对象,相比 Substring 或数组切片性能高且无 GC 影响。

堆上切片Memory 、 ReadOnlyMemory

特点和用法类似于Span,不同点时可以分配到堆上

csharp 复制代码
async Task<int> ReadStreamAsync(Stream stream)
{
    Memory<byte> buffer = new byte[1024];
    int bytesRead = await stream.ReadAsync(buffer);
    return bytesRead;
}

上面返回的是Memory类型,传统的方式是返回byte[],但是byte[] 可能导致异步复制,Memory 可直接传递,减少内存分配

在实际开发中,Span其实可以和Memory根据需求进行转换,比如Memory转换成span
Span<T> span = buffer.Span

同样,也可以把span转换为Memory
Memory<T> m = MemoryMarshal.AsMemory(span);

相关推荐
步步为营DotNet1 天前
深入理解IAsyncEnumerable:.NET中的异步迭代利器
服务器·前端·.net
玩泥巴的1 天前
强的飞起的 Roslyn 编译时代码生成,实现抽象类继承与依赖注入的自动化配置
c#·.net·代码生成·roslyn
mudtools1 天前
强的飞起的 Roslyn 编译时代码生成,实现抽象类继承与依赖注入的自动化配置
c#·.net
马达加斯加D1 天前
Web框架 --- .Net中的Worker Service
.net
步步为营DotNet1 天前
深入解读CancellationToken:.NET异步操作的精准控制
java·前端·.net
运维小文1 天前
Centos7部署.net8和升级libstdc++
开发语言·c++·.net
步步为营DotNet1 天前
深度探究.NET中的IAsyncEnumerable:异步迭代的底层奥秘与高效实践
.net
唐青枫2 天前
C# 泛型数学:解锁真正的类型安全数值运算
c#·.net
追逐时光者2 天前
C#/.NET/.NET Core技术前沿周刊 | 第 63 期(2025年11.24-11.30)
后端·.net