《每日一练》-实现一个单例模式

这篇博客探讨了C++中的单例模式实现,包括饿汉式和懒汉式的详细代码实现,并分析了它们的线程安全性和效率。同时,介绍了指令重排可能导致的单例陷阱,以及C++11通过静态内部变量解决这一问题的方法。文章最后提出,在C++11环境下,静态内部变量可以简化单例模式的实现并保证线程安全。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

单例模式

23种设计模式与面向对象

在这里插入图片描述

单例模式

饿汉式

/*************************************************************************
	> File Name: 1.singleton.cpp
	> Mail: 1136984246@qq.com
 ************************************************************************/
// 单例模式四要素(a-b)
#include<iostream>
using namespace std;
// 饿汉
class Singleton {
public:
    static Singleton* getInstance(); // 静态成员函数
private:
    // 构造函数必须私有,外部不可调用构造函数 --(a)--
    Singleton() {
        cout << "new";
    }
    Singleton(const Singleton&)=delete; // ?拷贝构造函数
    Singleton& operator=(const Singleton&)=delete; // ?赋值运算符?
    static Singleton* instance; // 静态私有成员变量 (全局唯一性,保证线程安全) --(b)--
};
// 饿汉标志:1.无需使用的时候再new,不支持延时加载,但是线程安全(在线程未创建时完成静态成员的初始化),调用时getInstance直接return;
// 不支持延迟加载 --(c)-- :有问题早发现,假如单例创建时有问题(OOM,内存泄漏),在创建时就报错,而不是使用时才报错
Singleton* Singleton::instance = new Singleton();
// getInstance在线程安全的情况下要效率高 --(d)--
Singleton* Singleton::getInstance() {
    return instance;
}
int main() {
    Singleton *s = Singleton::getInstance();
    Singleton *ss = Singleton::getInstance();
    Singleton *sss = Singleton::getInstance();
    return 0;
}

懒汉式

/*************************************************************************
	> File Name: 2.singleton.cpp
	> Mail: 1136984246@qq.com
 ************************************************************************/

#include<iostream>
#include <mutex>
using namespace std;

// 懒汉模式:简单加锁 -> 双重检测

class Singleton {
public:
    static Singleton* getInstance();
private:
    Singleton() {
        cout << "new" << endl;
    }
    Singleton(const Singleton&)=delete;
    Singleton& operator=(const Singleton)=delete;
    static Singleton* instance;
    static std::mutex m_mutex;
};

Singleton* Singleton::instance = nullptr; // ?
std::mutex Singleton::m_mutex;

// 双重检测
Singleton* Singleton::getInstance() {
    if (instance == nullptr) {
        std::lock_guard<std::mutex> lock(m_mutex); // ?
        if (instance == nullptr) {
            instance = new Singleton();
            // 指令重排:有一定概率,与CPU有关 -> 解决方案 -> java中的jvm,c#的volunteer,windows的c++ 
            // new非原子性操作 1.find找空间 2.alloc构造器 3.return (期待的方式)
            // 实际上:X86的优化:由于alloc费时,而且return之后,不一定直接使用 -> 先return后alloc
        }
    }
    return instance;
}

int main() {
    Singleton *s = Singleton::getInstance();
    Singleton *ss = Singleton::getInstance();
    Singleton *sss = Singleton::getInstance();
    return 0;
}

在这里插入图片描述在这里插入图片描述

指令重排与单例陷阱

对于指令重排:假如有线程A,B,A拿到锁,B阻塞,当A进行new操作,先find,再return;进行return instance先释放锁,过一段时间period再alloc;在这段period中,B拿到锁,再次做了一遍new;

c++11的静态内部变量

公司中本质上是C++98,要不用饿汉模式,要不就是复杂原子操作的懒汉模式。有c++11,直接用静态内部变量就可以了。

/*************************************************************************
	> File Name: 3.singleton.cpp
	> Mail: 1136984246@qq.com
 ************************************************************************/

#include<iostream>
using namespace std;
// 静态内部变量
class Singleton {    
public:
    ~Singleton() {
        cout << "out" << endl;
    }
    static Singleton& getInstance() {
        // 在内部new,实现了延迟加载
        static Singleton instance; // c++11 static 线程安全
        return instance;
    } 
private:
    Singleton(const Singleton&)=delete;
    Singleton& operator=(const Singleton&)=delete;
    Singleton() {
        cout << "new" << endl;
    }
};

int main() {
    Singleton &s = Singleton::getInstance();
    Singleton &ss = Singleton::getInstance();
    Singleton &sss = Singleton::getInstance();
    return 0;
}

C++本科毕业设计-ROS机械臂上位机源码(下载即用),个人经导师指导并认可通过的高分设计项目,评审分98分,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运行!主要针对计算机相关专业的正在做大作业、毕业设计的学生和需要项目实战练习的学习者,资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、使用需求,如果有需要的话可以放心下载使用。 C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源码(下载即用)C++本科毕业设计-ROS机械臂上位机源
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值