C++设计模式 - 单例模式(Singleton)

对象性能模式

面向对象很好地解决了"抽象”的问题,但是必不可免地要付出一定的代价。对于通常情况来讲,面向对象的成本大都可以忽略不计。但是某些情况,面向对象所带来的成本必须谨慎处理。

典型模式

Singleton

单例模式其意图是:保证一个类只有一个实例,并提供一个访问它的全局访问点。

  • 为了防止外部类构造实例,需要将构造函数的访问权限标记为protected或private;
  • 需要提供全局访问点,就需要在类中定义一个static函数,返回在类内部唯一构造的实例。

线程非安全版本

#include <iostream>

class Singleton{
private:
    Singleton(){
        std::cout<<"constructor called!"<<std::endl;
    }
    Singleton(Singleton&)=delete;
    Singleton& operator=(const Singleton&)=delete;
    static Singleton* m_instance_ptr;
public:
    ~Singleton(){
        std::cout<<"destructor called!"<<std::endl;
    }
    static Singleton* get_instance(){
        if(m_instance_ptr==nullptr){
              m_instance_ptr = new Singleton;
        }
        return m_instance_ptr;
    }
    void use() const { std::cout << "in use" << std::endl; }
};

Singleton* Singleton::m_instance_ptr = nullptr;

int main(){
    Singleton* instance = Singleton::get_instance();
    Singleton* instance_2 = Singleton::get_instance();
    return 0;
}

线程安全版本,但锁的代价过高

#include <iostream>
#include <mutex>

class Singleton
{
private:
    Singleton()
    {
        std::cout << "constructor called!" << std::endl;
    }
    Singleton(Singleton &) = delete;
    Singleton &operator=(const Singleton &) = delete;
    static Singleton *m_instance;
    static std::mutex m_mtx;

public:
    ~Singleton()
    {
        std::cout << "destructor called!" << std::endl;
    }
    static Singleton *get_instance()
    {
        std::lock_guard<std::mutex> lk{m_mtx};
        if (m_instance == nullptr)
        {
            m_instance = new Singleton();
        }
        return m_instance;
    }
};

Singleton *Singleton::m_instance = nullptr;
std::mutex Singleton::m_mtx;

int main()
{
    Singleton *instance = Singleton::get_instance();
    Singleton *instance_2 = Singleton::get_instance();
    return 0;
}

双检查锁,但由于内存读写reorder不安全

#include <iostream>
#include <mutex>

class Singleton
{
private:
    Singleton()
    {
        std::cout << "constructor called!" << std::endl;
    }
    Singleton(Singleton &) = delete;
    Singleton &operator=(const Singleton &) = delete;
    static Singleton *m_instance;
    static std::mutex m_mtx;

public:
    ~Singleton()
    {
        std::cout << "destructor called!" << std::endl;
    }
    static Singleton *get_instance()
    {
        if (m_instance == nullptr)
        {
            std::lock_guard<std::mutex> lk{m_mtx};
            if (m_instance == nullptr)
            {
                m_instance = new Singleton();
            }
        }
        return m_instance;
    }
};

Singleton *Singleton::m_instance = nullptr;
std::mutex Singleton::m_mtx;

int main()
{
    Singleton *instance = Singleton::get_instance();
    Singleton *instance_2 = Singleton::get_instance();
    return 0;
}

线程安全、内存安全的(智能指针,锁)

#include <iostream>
#include <memory>
#include <mutex>
class Singleton
{
public:
    typedef std::shared_ptr<Singleton> Ptr;
    ~Singleton()
    {
        std::cout << "destructor called!" << std::endl;
    }
    Singleton(Singleton &) = delete;
    Singleton &operator=(const Singleton &) = delete;
    static Ptr get_instance()
    {
        if (m_instance_ptr == nullptr)
        {
            std::lock_guard<std::mutex> lk(m_mutex);
            if (m_instance_ptr == nullptr)
            {
                m_instance_ptr = std::shared_ptr<Singleton>(new Singleton);
            }
        }
        return m_instance_ptr;
    }

private:
    Singleton()
    {
        std::cout << "constructor called!" << std::endl;
    }
    static Ptr m_instance_ptr;
    static std::mutex m_mutex;
};

Singleton::Ptr Singleton::m_instance_ptr = nullptr;
std::mutex Singleton::m_mutex;

int main()
{
    Singleton::Ptr instance = Singleton::get_instance();
    Singleton::Ptr instance2 = Singleton::get_instance();
    return 0;
}

使用静态局部变量安全又高效

#include <iostream>

class Singleton
{
public:
    ~Singleton()
    {
        std::cout << "destructor called!" << std::endl;
    }
    Singleton(const Singleton &) = delete;
    Singleton &operator=(const Singleton &) = delete;
    static Singleton &get_instance()
    {
        static Singleton instance;
        return instance;
    }

private:
    Singleton()
    {
        std::cout << "constructor called!" << std::endl;
    }
};

int main(int argc, char *argv[])
{
    Singleton &instance_1 = Singleton::get_instance();
    Singleton &instance_2 = Singleton::get_instance();
    return 0;
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吃米饭

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值