0

0

C++中如何正确实现双重检查锁定模式 现代C++内存模型下的单例优化方案

P粉602998670

P粉602998670

发布时间:2025-08-13 08:53:01

|

928人浏览过

|

来源于php中文网

原创

双重检查锁定用于减少加锁开销,只在首次初始化时加锁,后续访问无需进入临界区。1. 使用原子变量(std::atomic++)确保跨线程可见性;2. 通过memory_order_acquire和memory_order_release形成内存屏障,防止指令重排;3. 第一次检查非阻塞,第二次加锁确保构造完成后再写入指针。错误实现可能因重排导致读取未完全构造的对象。正确写法需结合原子操作与内存顺序,而推荐的替代方案是c++11静态局部变量方式,简洁且线程安全。

C++中如何正确实现双重检查锁定模式 现代C++内存模型下的单例优化方案

在C++中实现线程安全的单例模式时,双重检查锁定(Double-Checked Locking Pattern)是一个常见的优化手段。它旨在减少加锁带来的性能开销,只在第一次初始化时加锁,后续访问无需再进入临界区。

C++中如何正确实现双重检查锁定模式 现代C++内存模型下的单例优化方案

但很多人写出来的“双重检查”其实并不正确,尤其是在现代C++内存模型下,如果没有正确使用内存顺序或原子操作,可能会导致未定义行为,比如读取到未完全构造的对象。

下面我们就来看看如何在现代C++中正确地实现这个模式。

立即学习C++免费学习笔记(深入)”;

C++中如何正确实现双重检查锁定模式 现代C++内存模型下的单例优化方案

为什么需要双重检查锁定?

通常我们会用静态局部变量来实现懒加载的单例,比如这样:

MyClass& GetInstance() {
    static MyClass instance;
    return instance;
}

这种方式简洁且线程安全(C++11起),但在某些场景下不够灵活,比如我们希望控制实例的创建时机,或者想延迟加载以提升启动速度。

C++中如何正确实现双重检查锁定模式 现代C++内存模型下的单例优化方案

这时候就会想到手动管理单例对象,结合互斥锁来保证线程安全:

std::mutex mtx;

MyClass* GetInstance() {
    std::lock_guard lock(mtx);
    if (!instance) {
        instance = new MyClass();
    }
    return instance;
}

问题是:每次调用都要加锁,效率低。于是就引出了双重检查锁定,它的核心思想是:只在第一次创建对象时加锁,之后不再加锁


正确实现双重检查锁定的关键点

要正确实现双重检查锁定,需要注意以下几点:

Pic Copilot
Pic Copilot

AI时代的顶级电商设计师,轻松打造爆款产品图片

下载
  • 使用原子变量确保可见性
  • 防止指令重排影响对象构造顺序
  • 合理使用内存顺序避免过度同步

下面是常见但错误的写法:

MyClass* ptr = nullptr;

MyClass* GetInstance() {
    if (!ptr) {                     // 第一次检查
        std::lock_guard lock(mtx);
        if (!ptr) {                 // 第二次检查
            ptr = new MyClass();    // 潜在问题在这里
        }
    }
    return ptr;
}

问题出在

ptr = new MyClass()
这行代码上。虽然看起来是一条语句,但实际上分为三个步骤:

  1. 分配内存
  2. 调用构造函数
  3. 将指针赋值给
    ptr

由于编译器或CPU可能对指令进行重排优化,在多线程环境下可能导致其他线程看到一个已经分配内存但尚未构造完成的对象。


如何修正?使用原子变量 + 内存顺序

正确的做法是将

ptr
声明为
std::atomic
,并配合合适的内存顺序:

std::atomic ptr(nullptr);
std::mutex mtx;

MyClass* GetInstance() {
    MyClass* p = ptr.load(std::memory_order_acquire);
    if (!p) {
        std::lock_guard lock(mtx);
        p = ptr.load(std::memory_order_relaxed);
        if (!p) {
            p = new MyClass();
            ptr.store(p, std::memory_order_release);  // 确保构造完成后再写入
        }
    }
    return p;
}

这里有几个关键点:

  • std::atomic
    保证了跨线程的可见性
  • memory_order_acquire
    memory_order_release
    形成内存屏障,防止构造过程被重排序
  • 第一次检查是非阻塞的,第二次才真正加锁

替代方案:更简单的办法

如果你只是想要一个线程安全的单例,并不关心具体实现细节,推荐使用C++11标准提供的静态局部变量方式:

MyClass& GetInstance() {
    static MyClass instance;
    return instance;
}

这是最简洁、最安全的做法,而且现代编译器都做了很好的优化。

只有当你确实需要手动控制生命周期或延迟加载时,才考虑双重检查锁定这种复杂模式。


基本上就这些。双重检查锁定看起来简单,但要写对并不容易,尤其是在涉及并发和内存模型的时候。理解背后原理,才能写出高效又安全的代码。

相关专题

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

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

52

2025.08.29

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

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

98

2025.10.23

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

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

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

34

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

14

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

33

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

18

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.13

热门下载

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

精品课程

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

共94课时 | 6.7万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.3万人学习

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

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