C++实现单例模式及遇到的问题

本文介绍了C++中单例模式的三种实现方式:懒汉式、饿汉式和双重检验,并详细讨论了在实现过程中遇到的问题,包括静态成员初始化的编译错误和new操作返回指针的误解。同时,强调了static修饰和构造函数私有化的重要性,以确保单例模式的正确应用。

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

通常而言,单例模式分为三种实现方式:

  1. 懒汉式:系统定义实例。
  2. 饿汉式:直接给用户定义一个实例,以后都是直接调用这个。
  3. 双重检验:通常用于多线程时线程安全的代码。

首先,展示一个错误的代码示范:(C++11)

class Singleton{
    public:
    static Singleton getInstance(){
        if(single) return single;
        else 
            single = new Singleton();
        
        cout<<"ADDR: "<<&single<<endl; // 对象,用取地址符输出地址,用于比较是否为同一个实例
        return single;
    }
    
    private:
    static Singleton single;
    Singleton()=default;
    ~Singleton()=default;
}

对于初学者来说,咋一看可能没觉得有问题,其实有两个严重问题,如下:

 问题1: 

 if(single) return single

编译器会告诉你: 

could not convert ‘Singleton::single’ from ‘Singleton’ to ‘bool’

一般人会认为,对象没创建时肯定是空,也就是bool类型的false,好像可以用if直接判断。 显然编译器已经告诉了我们答案:绝对不可以!

 准确来说,Singleton::single是一个未识别的类型,因为是你自己定义的,除非你自己通过重载运算符之类的定义了隐式转换,否则,编译器无法为你进行隐式转换成bool类型。

问题2:

            single = new Singleton();

new产生一个新的对象,返回的不是新的对象!而是指向这个新对像的指针!

可以看一下new的底层调用形式:

void * operator new(size_t size)

很明显,返回的是指针。所以,用来接收收新对象时须定义一个指针。

补充:

  • 这里使用static修饰实例对象是因为,单例,即这个类只能存在这一个实例,显然,static符合。
  • 把构造函数和析构函数私有化,就是为了防止用户直接调用构造函数,破坏单例。

下面,开始贴上正确代码:

1. 懒汉式

#include <iostream>

using namespace std;

class Singleton{
    public:
    static Singleton* getInstance(){
        if(single== nullptr)  
            single = new Singleton();
        
        cout<<"ADDR: "<<single<<endl;  // &single替换成single,因为single本身存的就是地址了
        return single;
    }
    
    private:
    static Singleton *single;
    Singleton()=default;
    ~Singleton()=default;
};
 Singleton* Singleton::single = nullptr;
int main()
{
   Singleton* s;
   s->getInstance();
   Singleton* s2;
   s2->getInstance();
   Singleton* s3;
   s3->getInstance();
   return 0;
}

输出:

$g++ -std=c++11 -o main *.cpp
$main
ADDR: 0x601178
ADDR: 0x601178
ADDR: 0x601178

2. 饿汉式

#include <iostream>

using namespace std;

class Singleton{
    public:
    static Singleton* getInstance(){
        if(single== nullptr)  
            single = new Singleton();
        
        cout<<"ADDR: "<<single<<endl;
        return single;
    }
    void destroy(){
        cout<<"Invoke deconstructor."<<endl;
        delete single; //销毁对象,但指针仍指着那块地址。
        single = nullptr; //重置指针
    }
    
    private:
    static Singleton *single;
    Singleton()=default;
    ~Singleton()=default;
};
 Singleton* Singleton::single = new Singleton() ; // 直接实例化
int main()
{
   Singleton* s=Singleton::getInstance();
   s->destroy();

   Singleton* s2=Singleton::getInstance();
   s2->destroy();

   Singleton* s3=Singleton::getInstance();
   
   return 0;
}

对应的输出:

$g++ -std=c++11 -o main *.cpp
$main
ADDR: 0x991c20
Invoke deconstructor.
ADDR: 0x991c20
Invoke deconstructor.
ADDR: 0x991c20

3. 双重校验--用于多线程甚至多进程环境。

#include <iostream>
#include<pthread.h>
using namespace std;
class Singleton{
    public:
    static Singleton* getInstance(){
        if(single== nullptr)  {
            pthread_mutex_lock(&mutex_);
            if(single== nullptr)
                 single = new Singleton();
            pthread_mutex_unlock(&mutex_);
        }
        cout<<"ADDR: "<<single<<endl;
        return single;
    }
    void destroy(){
        cout<<"Invoke deconstructor."<<endl;
        delete single; //销毁对象,但指针仍指着那块地址。
        single = nullptr; //重置指针
    }
    
    private:
    static Singleton *single;
    static pthread_mutex_t mutex_;
    Singleton()=default;
    ~Singleton()=default;
};
 Singleton* Singleton::single = nullptr ; // 直接实例化
 pthread_mutex_t Singleton::mutex_;
 void gg(){
     Singleton::getInstance();
 }
int main()
{
   
    pthread_t p1,p2;
    pthread_create (&p1,NULL,gg,NULL);
    pthread_create (&p2,NULL,gg,NULL);
   
   return 0;
}

作为进一步补充,可以参考一下网友的C++完美单例模式

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Poo_Chai

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

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

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

打赏作者

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

抵扣说明:

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

余额充值