单例模式

参考《大话设计模式》第21章

参考博客:https://www.cnblogs.com/cxjchen/p/3148582.html

https://www.cnblogs.com/qiaoconglovelife/p/5851163.htmlcankao 

单例模式(Singleton):保证一个类仅有一个实例,并提供一个访问它的全局访问点。

通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。一个最好的办法是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。

UML类图:


Singleton类,定义一个Getinstance操作,允许客户访问它的唯一实例。Getinstance是一个静态方法,主要负责创建自己的唯一实例。

class singleton
{
protected:
    singleton(){}
private:
    static singleton* p;
public:
    static singleton* instance();
};
singleton* singleton::p = NULL;
singleton* singleton::instance()
{
    if (p == NULL)
        p = new singleton();
    return p;
}

因为Singleton类封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它,简而言之:对唯一实例的受控访问。

多线程时:
很可能两个线程同时运行到if (instance == NULL)这一句,导致可能会产生两个实例。

通过加锁来实现,lock时确保当一个线程位于代码的临界区,另一个线程不进入临界区。如果其他线程试图进入锁定的代码,则它将一直等待(即被阻拦),直到该对象被释放。

由于这种方式每次调用Getinstance时都需要加锁,这样会影响性能,所以使用双重锁定,先判断是否存在实例,再加锁,再判断是否存在实例。(Double-Check Locking)

上述这种要在第一次被引用时,才会将自己实例化,所以被称为懒汉式单例类。(双重锁定保证安全)

C#与公共语言运行库提供了一种‘静态初始化方法’,不需要开发人员显式地编写线程安全代码,即可解决多环境下它是不安全的问题。这种静态初始化的方式是在自己被加载时就将自己实例化,称之为饿汉式单例类。(提前占用系统资源)

加锁的经典懒汉实现:

class singleton
{
protected:
    singleton()
    {
        pthread_mutex_init(&mutex);
    }
private:
    static singleton* p;
public:
    static pthread_mutex_t mutex;
    static singleton* initance();
};

pthread_mutex_t singleton::mutex;
singleton* singleton::p = NULL;
singleton* singleton::initance()
{
    if (p == NULL)
    {
        pthread_mutex_lock(&mutex);
        if (p == NULL)
            p = new singleton();
        pthread_mutex_unlock(&mutex);
    }
    return p;
}

内部静态变量的懒汉实现:

class singleton
{
protected:
    singleton()
    {
        pthread_mutex_init(&mutex);
    }
public:
    static pthread_mutex_t mutex;
    static singleton* initance();
    int a;
};

pthread_mutex_t singleton::mutex;
singleton* singleton::initance()
{
    pthread_mutex_lock(&mutex);
    static singleton obj;
    pthread_mutex_unlock(&mutex);
    return &obj;
}

饿汉实现:

class singleton
{
protected:
    singleton()
    {}
private:
    static singleton* p;
public:
    static singleton* initance();
};
singleton* singleton::p = new singleton;
singleton* singleton::initance()
{
    return p;
}

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值