0

0

c# C#中的双重检查锁定(Double-Checked Locking)为什么是错的(在没有volatile时)

幻夢星雲

幻夢星雲

发布时间:2025-12-31 09:34:02

|

650人浏览过

|

来源于php中文网

原创

加 volatile 可修复双重检查锁定错误,因其禁止对 _instance 的读写重排序并确保内存可见性;推荐改用 Lazy 或静态构造函数实现线程安全单例。

c# c#中的双重检查锁定(double-checked locking)为什么是错的(在没有volatile时)

为什么没有 volatile 的双重检查锁定会出错

因为在 .NET 2.0+ 的 JIT 编译器和 x86/x64 内存模型下,new Singleton() 的执行可能被重排序:分配内存 → 写入字段 → 调用构造函数。如果线程 A 在构造函数尚未完成时,就将已分配但未初始化完毕的 _instance 引用写回主内存(或让其他线程看到),线程 B 就可能拿到一个「半初始化」的对象——调用其方法时抛出 NullReferenceException 或更隐蔽的逻辑错误。

Double-Checked Locking 的典型错误写法长什么样

下面这段代码在 .NET Framework 2.0–4.7(未加 volatile)和早期 .NET Core 版本中是不安全的:

public sealed class Singleton
{
    private static 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;
        }
    }

    private Singleton() { }
}

问题不在 lock,而在于:lock 只保证临界区的互斥,不保证对 _instance 的写操作对其他线程「立即可见」,也不阻止 JIT 将对象初始化步骤重排。

为什么加 volatile 就能修好

volatile_instance 字段施加了两个关键约束:

  • 禁止编译器和 JIT 对该字段的读/写进行重排序(特别是禁止把 _instance = new Singleton() 中的引用赋值提前到构造函数执行完之前)
  • 确保每次读取都从主内存(或最新缓存)获取值,每次写入都立即刷新到主内存,使其他线程能及时看到更新

修正后的安全写法:

SlidesAI
SlidesAI

使用SlidesAI的AI在几秒钟内创建演示文稿幻灯片

下载
public sealed class Singleton
{
    private static volatile Singleton _instance; // ✅ 加 volatile
    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;
        }
    }

    private Singleton() { }
}

但其实你根本不需要手写双重检查锁定

.NET 已提供更简洁、更可靠的方式:

  • Lazy:默认线程安全,内部使用正确的内存屏障,且支持延迟初始化 + 异常缓存
  • 用静态构造函数:CLR 保证只执行一次且完全线程安全,但不支持懒加载(类型首次被访问即初始化)

推荐写法(懒加载 + 安全 + 简洁):

public sealed class Singleton
{
    private static readonly Lazy _lazy =
        new Lazy(() => new Singleton());

    public static Singleton Instance => _lazy.Value;

    private Singleton() { }
}

真正容易被忽略的是:即使加了 volatile,双重检查锁定仍比 Lazy 更难验证、更易误用(比如漏掉 volatile、改用非引用类型、或在构造函数里暴露 this)。除非你在极老的 .NET 版本上无法用 Lazy,否则别碰它。

相关专题

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

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

48

2025.08.29

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

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

95

2025.10.23

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

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

66

2025.10.23

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

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

469

2023.08.10

excel制作动态图表教程
excel制作动态图表教程

本专题整合了excel制作动态图表相关教程,阅读专题下面的文章了解更多详细教程。

24

2025.12.29

freeok看剧入口合集
freeok看剧入口合集

本专题整合了freeok看剧入口网址,阅读下面的文章了解更多网址。

74

2025.12.29

俄罗斯搜索引擎Yandex最新官方入口网址
俄罗斯搜索引擎Yandex最新官方入口网址

Yandex官方入口网址是https://yandex.com;用户可通过网页端直连或移动端浏览器直接访问,无需登录即可使用搜索、图片、新闻、地图等全部基础功能,并支持多语种检索与静态资源精准筛选。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

207

2025.12.29

python中def的用法大全
python中def的用法大全

def关键字用于在Python中定义函数。其基本语法包括函数名、参数列表、文档字符串和返回值。使用def可以定义无参数、单参数、多参数、默认参数和可变参数的函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

16

2025.12.29

python改成中文版教程大全
python改成中文版教程大全

Python界面可通过以下方法改为中文版:修改系统语言环境:更改系统语言为“中文(简体)”。使用 IDE 修改:在 PyCharm 等 IDE 中更改语言设置为“中文”。使用 IDLE 修改:在 IDLE 中修改语言为“Chinese”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

18

2025.12.29

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
nginx浅谈
nginx浅谈

共15课时 | 0.8万人学习

前端开发(基础+实战项目合集)
前端开发(基础+实战项目合集)

共60课时 | 3.7万人学习

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

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