C#怎么使用ValueTask C# ValueTask与Task性能对比

来源:这里教程网 时间:2026-02-21 17:33:46 作者:

ValueTask 是什么,什么时候该用它

ValueTask 是 C# 7.0 引入的轻量级结构体(struct),用来替代部分 Task 场景,避免不必要的堆分配。它内部可以包装一个 Task(异步未完成时)或直接保存同步结果(如

TResult
void
)。本质是“可选堆分配”的异步操作容器。

适合用 ValueTask 的典型场景:

方法大概率同步完成(比如缓存命中、内存数据快速返回) 被高频调用(如 ASP.NET Core 中间件、序列化器、基础工具方法) 不需多次 await(ValueTask 不可重复 await,重复调用会抛异常) 不需作为 Task 对象传递给其他 API(比如不能传给
Task.WhenAll
,得先
.AsTask()

怎么正确使用 ValueTask

声明和返回很简单,但要注意约束:

方法签名用
ValueTask<t></t>
ValueTask
(对应 void)
同步路径直接 return new ValueTask(value),不 new Task 异步路径用
await
普通 async 方法,编译器自动构造 ValueTask
需要转成 Task 时调用
.AsTask()
(仅在必须兼容 Task API 时才做)

示例:

public ValueTask<string> ReadAsync()
{
    if (_cache != null) // 同步命中
        return new ValueTask<string>(_cache);
<pre class='brush:php;toolbar:false;'>return ReadFromNetworkAsync(); // 真正 async 方法,返回 ValueTask<string>

}

private async ValueTask ReadFromNetworkAsync() { await Task.Delay(100); // 模拟 I/O return "data"; }

ValueTask 和 Task 性能差异在哪

核心区别在内存分配

Task 每次都分配在堆上(哪怕同步完成),GC 压力随调用频次上升 ValueTask 同步路径零分配(纯栈/寄存器),异步路径仍分配一次 Task,但结构体本身仍为栈值

实际性能提升取决于使用模式:

100% 同步:ValueTask 几乎无开销,Task 多一次 GC 友好但可测的堆分配 50% 同步 + 50% 异步:ValueTask 平均减少约 30–50% 的短期对象分配 100% 异步:两者表现接近(都需 Task 分配),ValueTask 额外有 struct 拷贝成本,但通常可忽略

别只看微基准——高吞吐服务(如 JSON 序列化、路由匹配)中,把关键路径从 Task 改成 ValueTask,常能降低 Gen0 GC 次数 10%+。

不该用 ValueTask 的情况

滥用反而有害:

方法总是异步(比如固定要读文件、发 HTTP 请求)→ 用 Task 更清晰、更安全 需要多次 await 同一个对象 → ValueTask 不支持,会崩溃 要参与 Task 组合(
WhenAll
WhenAny
ContinueWith
)→ 必须先
.AsTask()
,失去优势
公开 API 设计阶段不确定调用方用途 → 优先用 Task,保证兼容性和语义明确

一句话:ValueTask 是性能优化手段,不是 Task 的通用替代品。

基本上就这些。用对地方能省点分配,用错地方反而添麻烦。

相关推荐