0

0

c# 内存屏障(Memory Barrier)在C#中的作用和具体实现

月夜之吻

月夜之吻

发布时间:2026-01-29 12:40:04

|

485人浏览过

|

来源于php中文网

原创

内存屏障解决多线程下指令重排与内存可见性问题,尤其在弱内存模型CPU上防止读写乱序导致状态不一致。

c# 内存屏障(memory barrier)在c#中的作用和具体实现

内存屏障解决什么问题

多线程环境下,volatile 修饰的字段只能防止编译器重排序和部分 CPU 重排序,但无法阻止所有指令重排,尤其在弱内存模型 CPU(如 ARM、ARM64)上,读写操作可能被乱序执行,导致其他线程看到不一致的状态。内存屏障就是用来显式插入同步点,强制约束指令执行顺序和内存可见性。

MemoryBarrier 和 Thread.MemoryBarrier 的区别

Thread.MemoryBarrier() 是 .NET Framework 2.0 引入的全屏障(full fence),它同时禁止编译器和 CPU 的读写重排:前面的读写不能移到屏障后,后面的读写也不能移到屏障前。而 Thread.MemoryBarrier() 在 .NET Core / .NET 5+ 中已被标记为过时,推荐使用更明确的 Thread.VolatileRead() / Thread.VolatileWrite()Interlocked 系列操作。

真正底层、仍在使用的屏障是 Thread.MemoryBarrier() 对应的 IL 指令 monitorenter / monitorexit 隐含的语义,以及 Interlocked 方法内部隐含的屏障——例如 Interlocked.CompareExchange() 在 x86 上会生成 lock cmpxchg,天然带全屏障;在 ARM64 上则会插入 dmb ish 指令。

  • Thread.MemoryBarrier():已过时,仅用于兼容旧代码,不推荐新项目使用
  • Thread.VolatileRead(ref int) / Thread.VolatileWrite(ref int, int):分别提供 acquire-load 和 release-store 语义,比全屏障轻量
  • Interlocked.*():如 Interlocked.Increment(),不仅原子,还自带 full barrier,适合需要修改+同步的场景

volatile 字段 + 内存屏障的典型误用

很多人以为给字段加 volatile 就能安全实现双检锁(Double-Check Locking),但这是错的。比如下面这个单例模式片段:

private static volatile Singleton _instance;
private static readonly object _lock = new object();

public static Singleton Instance { get { if (_instance == null) { lock (_lock) { if (_instance == null) { _instance = new Singleton(); // ⚠️ 构造函数可能被重排到赋值之后! } } } return _instance; } }

问题在于:即使 _instancevolatile,C# 编译器和 CPU 仍可能把 new Singleton() 的三步(分配内存 → 调用构造函数 → 赋值给 _instance)重排成「分配 → 赋值 → 构造」,导致其他线程拿到一个未完全初始化的对象。

Draft&Goal-Detector
Draft&Goal-Detector

检测文本是由 AI 还是人类编写的

下载

正确做法是用 Interlocked.CompareExchange() 或 C# 6+ 的 Lazy(其内部使用了正确的屏障):

private static readonly Lazy _lazy =
    new Lazy(() => new Singleton(), isThreadSafe: true);

public static Singleton Instance => _lazy.Value;

何时该手动插入屏障(极少需要)

绝大多数情况你不需要手写 Thread.MemoryBarrier()。现代 C# 提供了更高层、更安全的抽象:

  • Interlocked 替代自增/比较交换
  • ConcurrentQueue / ConcurrentDictionary 替代手动加锁+屏障
  • ManualResetEventSlimSpinWait 替代忙等+裸屏障
  • 只有在极少数性能敏感、且必须绕过 .NET 同步原语(如实现无锁队列)时,才需结合 UnsafeInterlocked 和平台相关屏障(如 Thread.MemoryBarrier()Atomic.Read/Write 在 .NET 8+)

真正容易被忽略的是:ARM64 上 volatile 的语义比 x64 更弱,仅保证单个读/写不被重排,不保证前后访存顺序——这意味着依赖 volatile 实现状态协同的代码,在跨平台部署时可能突然出错。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

93

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

200

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

93

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

102

2025.10.23

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

69

2025.10.23

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

503

2023.08.10

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

19

2026.01.29

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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