c# Thread.VolatileRead 和 Thread.VolatileWrite 的用法

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

Thread.VolatileRead 和 Thread.VolatileWrite 是干什么的

它们用于对字段执行**带内存屏障的读/写操作**,确保当前线程看到的是最新值(不被 CPU 缓存或编译器重排序干扰),但不提供原子性保证(比如

long
double
的 64 位读写在 32 位系统上仍可能撕裂)。它们适用于极简场景:单个字段、无锁、且只需防止指令重排和缓存不一致——比如一个标志位的快速通知。

什么时候该用 VolatileRead/VolatileWrite,而不是 volatile 字段

volatile
关键字修饰字段时,C# 编译器会自动为所有对该字段的读写插入等效的
Thread.VolatileRead
/
Thread.VolatileWrite
语义。所以绝大多数情况下,直接声明
private volatile bool _isRunning;
就够了,更简洁、不易出错。

只有两个典型例外需要显式调用:

字段是
ref
或通过指针访问(比如
ref int
参数),无法加
volatile
修饰符
你正在写泛型或 unsafe 代码,需要对任意地址做 volatile 访问(例如在自定义同步原语中)

怎么正确使用 Thread.VolatileRead 和 Thread.VolatileWrite

必须传入字段的地址(

ref
),且类型只能是
bool
byte
char
short
int
long
sbyte
ushort
uint
ulong
float
double
或引用类型。不能用于 struct(哪怕只含一个 int)。

常见错误包括:

对非字段变量(如局部变量)取 ref 并传给 VolatileRead —— 编译失败或运行时异常 误以为它能保证复合操作(如
if (_flag) { _flag = false; }
)的线程安全 —— 它不能,这是竞态条件
在 .NET Core 3.0+ 中,对
long
double
使用 VolatileRead/VolatileWrite 仍可能因硬件不支持而降级为普通读写(尤其 ARM64)
private int _counter;
private void Increment()
{
    // ✅ 正确:对字段 ref 操作
    Thread.VolatileWrite(ref _counter, Thread.VolatileRead(ref _counter) + 1);
<pre class='brush:php;toolbar:false;'>// ❌ 错误:局部变量不能这样用
// int local = 0;
// Thread.VolatileWrite(ref local, 1); // 编译报错

}

比 volatile 更强的替代方案有哪些

如果你需要真正线程安全的读-改-写,或者跨平台强一致性,优先考虑:

Interlocked.CompareExchange
:原子地比较并交换,适合实现无锁栈、计数器、状态机
MemoryBarrier
(或
Thread.MemoryBarrier
):手动插入全屏障,粒度更细但易误用
C# 7.3+ 的
ref readonly
+
Unsafe
类型:仅限极端性能场景,需
unsafe
上下文

注意:

Thread.VolatileRead
Thread.VolatileWrite
在 .NET 5+ 已标记为 [Obsolete],官方建议迁移到
System.Threading.Volatile
类的静态方法(
Volatile.Read
/
Volatile.Write
),语义一致但更现代、支持泛型重载。

相关推荐