在多线程环境下,单例模式需要保证全局只有一个实例且初始化过程线程安全。传统的双重检查锁定加锁方式会引入锁竞争带来的性能损耗,而无锁实现通过atomic原子变量和内存屏障的配合,可以在无锁的情况下保证单例初始化的安全性。

单例模式的基础要求
单例模式需要满足三个核心要求:一是全局只有一个实例,二是实例的初始化过程线程安全,三是提供全局统一的访问入口。在并发场景下,最常见的风险是多个线程同时触发实例初始化,导致创建多个实例,或者出现未完全初始化的实例被其他线程使用的情况。
atomic原子变量的作用
C++11引入的std::atomic模板类提供了原子操作能力,对原子变量的读写操作不会被线程调度机制打断,能保证操作的可见性和原子性。在实现无锁单例时,我们通常会用一个原子变量来标记实例是否已经初始化完成,避免多个线程同时进入初始化逻辑。
需要注意的是,普通的布尔变量或者指针变量在多线程读写时可能出现可见性问题,一个线程对变量的修改可能不会立即被其他线程感知,而原子变量通过底层的内存序设置可以解决这个问题。
内存屏障的必要性
编译器和CPU为了优化性能,可能会对指令进行重排。在单例初始化场景中,可能出现的情况是:先分配了内存地址赋值给实例指针,但是实例的构造函数还没有执行完成,此时其他线程读取到非空的指针就会返回一个未初始化完成的实例,导致程序错误。
内存屏障可以阻止特定类型的指令重排,保证屏障前后的指令执行顺序符合代码逻辑。C++的atomic操作可以通过指定内存序参数来隐式插入内存屏障,不需要手动编写平台相关的屏障指令。
无锁单例完整实现源码
以下是基于C++11标准实现的无锁并发单例模式完整代码,使用了atomic原子变量和正确的内存序设置:
#include <atomic>
#include <iostream>
// 单例类定义
class Singleton {
private:
// 私有构造函数,防止外部直接实例化
Singleton() {
std::cout << "Singleton instance initialized" << std::endl;
}
// 禁止拷贝构造和赋值操作
Singleton(const Singleton&) = delete;
Singleton& operator=(const Singleton&) = delete;
public:
// 全局访问入口
static Singleton* getInstance() {
// 第一次检查,使用内存序memory_order_acquire,保证后续操作能看到其他线程的写入
Singleton* tmp = instance.load(std::memory_order_acquire);
if (tmp == nullptr) {
// 加锁保护初始化过程,这里只是为了初始化临界区,不是无锁的核心
// 实际无锁实现可以结合call_once,不过这里展示原子变量+内存屏障的逻辑
std::call_once(init_flag, []() {
// 创建实例,此时内存序保证构造完成后再赋值指针
Singleton* new_instance = new Singleton();
// 赋值操作使用memory_order_release,保证构造操作在赋值前完成,且对其他线程可见
instance.store(new_instance, std::memory_order_release);
});
tmp = instance.load(std::memory_order_acquire);
}
return tmp;
}
// 示例业务方法
void doSomething() {
std::cout << "Singleton do something" << std::endl;
}
// 释放实例的方法,实际场景中可以根据需求决定是否提供
static void destroyInstance() {
Singleton* tmp = instance.load(std::memory_order_acquire);
if (tmp != nullptr) {
delete tmp;
instance.store(nullptr, std::memory_order_release);
}
}
private:
// 原子指针变量,存储单例实例地址
static std::atomic<Singleton*> instance;
// call_once的标志变量
static std::once_flag init_flag;
};
// 静态成员初始化
std::atomic<Singleton*> Singleton::instance(nullptr);
std::once_flag Singleton::init_flag;
// 测试代码
int main() {
// 多线程获取单例实例
Singleton* s1 = Singleton::getInstance();
Singleton* s2 = Singleton::getInstance();
std::cout << "s1 address: " << s1 << std::endl;
std::cout << "s2 address: " << s2 << std::endl;
if (s1 == s2) {
std::cout << "Two instances are the same, singleton works" << std::endl;
}
s1->doSomething();
// 销毁实例
Singleton::destroyInstance();
return 0;
}
代码逻辑解析
上述代码中,instance是原子指针变量,初始值为nullptr。在getInstance方法中,首先用memory_order_acquire内存序加载实例指针,这个内存序保证当前线程能看到其他线程用memory_order_release写入的值,避免出现读取到旧值的情况。
初始化过程使用std::call_once保证初始化逻辑只执行一次,在初始化逻辑内部,先执行Singleton的构造函数,再将实例地址存储到原子变量中,存储操作使用memory_order_release内存序,保证构造函数中的所有操作在指针赋值前完成,并且对其他线程可见,避免指令重排导致的未初始化实例被使用的问题。
无锁单例的适用场景
无锁并发单例适合实例初始化频率不高、对性能要求较高的场景。如果单例初始化过程非常复杂或者耗时很长,仍然可能带来短暂的初始化竞争,此时可以根据实际情况选择是否结合轻量级的同步机制。另外需要注意,原子操作的内存序设置需要符合逻辑,错误设置可能导致可见性问题或者指令重排问题,实际使用时需要根据场景选择合适的内存序参数。
C++无锁并发单例atomic原子变量内存屏障修改时间:2026-07-04 18:00:16