确保对象的唯一性 - 单例模式

本文详细介绍了单例模式的概念、作用以及在C++中的实现方式,包括简单的单例模式、饿汉式单例模式和使用互斥锁的单例模式。通过实例分析了多线程环境下可能遇到的问题及解决方案,帮助读者掌握如何确保对象的唯一性并优化性能。

    单例模式在我们的日常开发中用的非常多,单例模式最大的特点就是只能生成一个类对象。
    比方配置文件的类信息,就可以使用单例模式实现,因为配置文件是固定的,不论在代码的哪个位置获取配置项信息,得到的都应该是相同的信息。

    既然只能生成一个类对象,那么我们就得考虑下如何实现这个只生成一个对象,在C++中我们知道,当定义或new一个对象时,就会调用构造函数,构造函数的执行成功,才宣誓一个对象的诞生完成。也就是说构造函数的调用是随着对象的定义而发生的,所以要确保只能生成一个对象,就要是的构造函数不可随意调用。
    C++里,私有属性的成员函数或变量是不能被外部直接调用的,所以为了限制构造函数的调用,我们可以将构造函数设为私有,
取而代之的是定义一个公有的static静态函数与一个私有static静态类对象变量来实现单例模式。代码如下:

#include <iostream>
#include <string.h>
#include <stdio.h>

using namespace std;

class SigalMode
{
private:
    static SigalMode* m_pInstance;
    static int m_count;

    SigalMode()
    {
        cout << "SigalMode()" << endl;
        m_count++;
    }
    SigalMode(const SigalMode& obj);
    SigalMode& operator = (const SigalMode& obj);
public:

    static SigalMode* getInstance()  //单例模式主要代码块
    {
        if(NULL == m_pInstance)
        {
            m_pInstance = new SigalMode;
        }
        return m_pInstance;
    }

    int getCount()
    {
        return m_count;
    }

    ~SigalMode()
    {
        delete m_pInstance;
    }

};
int SigalMode::m_count = 0;
SigalMode* SigalMode::m_pInstance = NULL;

int main(int argc, char* argv[])
{
    cout << SigalMode::getInstance()->getCount() << endl;
    cout << SigalMode::getInstance()->getCount() << endl;
    cout << SigalMode::getInstance()->getCount() << endl;

    return 0;
}

编译执行:

从输出看到,我们调用的了三次GetInstance函数,但是构造函数只调用了一次,说明这个后两次GetInstance不调用构造函数,直接使用的是第一个GetInstance得到的类对象,这就是单例模式的简单实现。

    对于上边的单例模式,可以应对大部分的场景,但是还不够,假如我们多线程使用同一个单例,那么在GetIstance函数里就有可能出现:A线程判判断m_pInstance为NULL,进行new操作,由与A线程还未new完成,此时B线程执行到这步,也会得到m_pInstance为NULL,从而B线程也会进行new操作,那么此时A、B线程操作的就不是同一个对象。

    对于多线程引发的问题,我们可以按如下两种方式解决:
    第一种方式:在类定义是就将单例对象实例化(饿汉式单例模式)
    第二种方式:使用互斥锁

    先来看第一种方式:类定时时单例对象实例化,也就是说在类定义完成时,我们的单例对象就已经产生了

#include <iostream>
#include <string.h>
#include <errno.h>
#include <stdio.h>

using namespace std;

class SigalMode
{
private:
    static SigalMode* m_pInstance;
    static int m_count;

    SigalMode()
    {
        cout << "SigalMode()" << endl;
        m_count++;
    }
    SigalMode(const SigalMode& obj);
    SigalMode& operator = (const SigalMode& obj);
public:

    static SigalMode* getInstance()
    {
        return m_pInstance; //直接返回实例化后的对象
    }

    int getCount()
    {
        return m_count;
    }

    ~SigalMode()
    {
        delete m_pInstance;
    }

};
int SigalMode::m_count = 0;
SigalMode* SigalMode::m_pInstance = new SigalMode; //实例化单例对象



int main(int argc, char* argv[])
{

    cout << SigalMode::getInstance()->getCount() << endl;
    cout << SigalMode::getInstance()->getCount() << endl;
    cout << SigalMode::getInstance()->getCount() << endl;

    return 0;
}

编译执行

第一种方式虽然能够解决问题,但是代价是无论是不是用单例对象,这个对象从一开始就被创建,某种程度来说,浪费了内存空间。

第二种方式互斥锁
    使用互斥锁,主要点就是在GetInstance函数里把代码锁住,下边只给出这部分的代码,其余代码都一样

    static SigalMode* getInstance()
    {
        pthread_mutex_lock(&mutex_lock); //加锁
        if(NULL == m_pInstance)
        {
            m_pInstance = new SigalMode;
        }
        pthread_mutex_unlock(&mutex_lock); //解锁
        return m_pInstance;       
    }

    这么使用锁之后,问题就可以解决了,但是还是有点问题,单例模式中,只需要调用一次new,后边就不需要new了,但是多线程使用时每次都需要等待锁的释放,这影响了执行效率。下边来看另外一种加使用锁的方式(使用双重锁):

static SigalMode* getInstance()
{
    if(NULL == m_pInstance)
    {
        pthread_mutex_lock(&mutex_lock);
        if(NULL == m_pInstance)
        {
            m_pInstance = new SigalMode;
        }
        pthread_mutex_unlock(&mutex_lock);
    }
    return m_pInstance;       
}

以上几种单例模式的实现方式,可根据具体情况具体选择。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值