0

0

c# volatile 关键字的作用和原理 c#内存模型

煙雲

煙雲

发布时间:2026-01-07 08:39:02

|

640人浏览过

|

来源于php中文网

原创

volatile 保证字段读写对其他线程立即可见,但不保证原子性;它通过插入 acquire/release 内存屏障防止重排序,适用于单写多读状态标志,不适用于计数器或复合操作。

c# volatile 关键字的作用和原理 c#内存模型

volatile 保证可见性,但不保证原子性

在 C# 多线程编程中,volatile 的核心作用是让一个字段的读写操作对其他线程“立即可见”。它不加锁、不阻塞,但也不能替代 lockInterlocked——比如对 volatile int countercounter++,仍然是非原子的,结果可能丢失。

  • 每次读 volatile 字段,都强制从主内存(或最新缓存行)加载,跳过线程本地寄存器缓存
  • 每次写 volatile 字段,都会立即刷新到主内存,并插入**写内存屏障(write barrier)**,防止该写操作被重排序到其后指令之前
  • 它不能阻止其他非 volatile 字段的重排序,也不能保证复合操作(如读-改-写)的完整性

volatile 如何阻止指令重排序?靠内存屏障

C# 编译器和 x86/x64 CPU 在生成代码时,默认可能把语句顺序优化调整。而 volatile 字段访问会隐式插入内存屏障(Memory Barrier),这是硬件/运行时层面的同步原语。

  • volatile 字段 → 插入 **acquire fence**:确保该读之后的所有读/写不被提前到它前面
  • volatile 字段 → 插入 **release fence**:确保该写之前的所有读/写不被延后到它后面
  • 这对实现“发布-订阅”模式很关键:比如用 volatile bool _ready 标记数据已就绪,能确保另一线程看到 _ready == true 时,也一定能看到此前所有对关联数据的写入
class ReadyExample
{
    private int _data = 0;
    private volatile bool _ready = false;
public void Publish()
{
    _data = 42;          // 普通写
    _ready = true;       // volatile 写 → release fence 插入此处
}

public int Consume()
{
    if (_ready)          // volatile 读 → acquire fence 插入此处
        return _data;    // 此时 _data 一定是 42,不会读到 0
    throw new InvalidOperationException();
}

}

哪些场景适合用 volatile?哪些绝对不行?

它只适用于极简的“状态标志”同步,不是通用并发工具

中小企业网站系统前台源码(SmallBusinessStarterKit)
中小企业网站系统前台源码(SmallBusinessStarterKit)

小型企业入门套件(The Small Business Starter Kit)提供了一个商业宣传网站的完整演示,他适合中小型企业。使用他创建的网站支持自定义模板,具有先进的功能,包括:内容和数据管理的SQL和XML数据源整合。该源码包含C#和VB两个版本,只有前台部分源码,微软官方截止到51aspx发布源码时还没有提供后台代码。小型企业网站入门套件的关键页面包括:产品分类显示新闻发布显示商户认证

下载
  • ✅ 合适:单写多读的布尔开关(如 volatile bool _stopping)、初始化完成标记、取消令牌(配合 CancellationToken 更推荐)
  • ❌ 不合适:计数器(counter++)、引用类型对象的深层状态变更、需要互斥访问的集合操作、任何涉及多个字段协同更新的逻辑
  • ⚠️ 注意:.NET 5+ 中 volatilelongdouble 在 32 位系统上仍需谨慎(虽已基本无问题,但历史兼容性提醒仍在文档中)

C# 内存模型下 volatile 的定位:轻量级可见性契约

C# 内存模型(CLI 规范 ECMA-335)规定:每个线程有自己视角的内存视图,而 volatile 是少数几个能跨线程“拉齐视角”的语言级机制之一。但它不建立 happens-before 关系的全序,也不提供锁那样的排他语义。

  • 它不等价于 Java 的 volatile(JMM 更严格),但在 .NET Core/.NET 5+ 上行为已高度一致
  • 底层依赖 JIT 编译器识别 volatile 并生成带 mov + mfence(x64)或 ldrex/strex(ARM)的指令序列
  • 不要试图用它绕过 lock 来提升性能——现代 lock 在无竞争时开销极低;滥用 volatile 反而因频繁内存屏障拖慢 CPU 流水线

真正容易被忽略的是:volatile 解决的是“我改了,你能不能马上看到”,而不是“我们能不能一起改”。只要涉及“改”本身需要同步,就必须换更重的机制。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

827

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

732

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

732

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16924

2023.08.03

C++ 高性能计算与并行编程
C++ 高性能计算与并行编程

本专题专注于 C++ 在高性能计算(HPC)与并行编程中的应用,涵盖多线程、并发数据处理、OpenMP、MPI、GPU加速等技术。通过实际案例,帮助开发者掌握 如何利用 C++ 进行大规模数据计算和并行处理,提高程序的执行效率,适应高性能计算与数据密集型应用场景。

1

2026.01.08

热门下载

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

精品课程

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

共23课时 | 2.3万人学习

C# 教程
C# 教程

共94课时 | 6.2万人学习

Java 教程
Java 教程

共578课时 | 43.3万人学习

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

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