『设计模式』单例模式

文章讨论了三种线程安全的单例模式实现方法:原始的懒汉模式、使用原子变量改进的懒汉模式以及局部静态变量版本。主要关注了如何在多线程环境下确保任务队列的实例只被创建一次。

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

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

#if 0
//饿汉模式
class TaskQueue
{
public:
    TaskQueue(const TaskQueue &t) = delete;
    TaskQueue& operator=(const TaskQueue &t) = delete;

    static TaskQueue* getInstance()
    {
        return m_tQueue;
    }

    void print()
    {
        cout << "task queue......." << endl;
    }
private:
    TaskQueue() = default;

    static TaskQueue *m_tQueue;
};
TaskQueue* TaskQueue::m_tQueue = new TaskQueue;

#else
// 懒汉模式
class TaskQueue
{
public:
    TaskQueue(const TaskQueue &t) = delete;
    TaskQueue &operator=(const TaskQueue &t) = delete;

    static TaskQueue *getInstance()
    {
        if (m_tQueue == nullptr) //双重检查锁定,可以多线程同时并行获取实例
        {
            m_mutex.lock();  //加互斥锁,可以防止多线程访问时,创建多个实例
            if (m_tQueue == nullptr)
            {
                m_tQueue = new TaskQueue;
            }
            m_mutex.unlock();
            
        }
        

        return m_tQueue;
    }

    void print()
    {
        cout << "task queue success create." << endl;
    }

private:
    TaskQueue() = default;

    static TaskQueue *m_tQueue;
    static mutex m_mutex;
};

TaskQueue *TaskQueue::m_tQueue = nullptr;
mutex TaskQueue::m_mutex;
#endif

int main(int argc, const char **argv)
{

    TaskQueue *task = TaskQueue::getInstance();
    task->print();
    return 0;
}

使用原子变量解决双重检查锁定的问题

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

// 懒汉模式
class TaskQueue
{
public:
    TaskQueue(const TaskQueue &t) = delete;
    TaskQueue &operator=(const TaskQueue &t) = delete;

    static TaskQueue *getInstance()
    {
        TaskQueue *task = m_ato_tQueue.load();
        if ( task == nullptr) 
        {
            m_mutex.lock(); 
            task = m_ato_tQueue.load(); //是为了多线程时,获取到锁的线程从原子里取出数据,然后再判断
            if (task == nullptr)
            {
                task = new TaskQueue;
                m_ato_tQueue.store(task);
            }
            m_mutex.unlock();
        }

        return task;
    }

    void print()
    {
        cout << "task queue success create." << endl;
    }

private:
    TaskQueue() = default;

    // static TaskQueue *m_tQueue;
    static atomic<TaskQueue*> m_ato_tQueue;
    static mutex m_mutex;
};

atomic<TaskQueue*> TaskQueue::m_ato_tQueue;
mutex TaskQueue::m_mutex;

int main(int argc, const char **argv)
{

    TaskQueue *task = TaskQueue::getInstance();
    task->print();
    return 0;
}

局部静态变量解决线程安全问题

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

// 懒汉模式
class TaskQueue
{
public:
    TaskQueue(const TaskQueue &t) = delete;
    TaskQueue &operator=(const TaskQueue &t) = delete;

    static TaskQueue *getInstance()
    {
        
        static TaskQueue tQueue; //局部取出变量
        return &tQueue;
    }

    void print()
    {
        cout << "task queue create successfully." << endl;
    }

private:
    TaskQueue() = default;

   
};


int main(int argc, const char **argv)
{

    TaskQueue *task = TaskQueue::getInstance();
    task->print();
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值