当前位置:首页 » 《休闲阅读》 » 正文

C++_单例模式

15 人参与  2024年10月15日 12:03  分类 : 《休闲阅读》  评论

点击全文阅读


目录

1、饿汉方式实现单例

2、懒汉方式实现单例

3、单例模式的总结 

结语


前言:

        在C++中有许多设计模式,单例模式就是其中的一种,该模式主要针对类而设计,确保在一个进程下该类只能实例化出一个对象,因此名为单例。而单例模式又分为饿汉方式和懒汉方式,饿汉方式指的是只要发出了对该类的需求,就会实例化对象。懒汉方式指的是即使发出了对该类的需求,但是不会立刻实例化对象,等到真正用到该对象时才会实例化对象。

        示意图如下:

1、饿汉方式实现单例

         饿汉方式表明只要提出需要,则立即用该类实例化对象,并且当前进程只能有一个该类型的对象。这个逻辑在代码中的体现是:只要我们提出需要某个类的实例化请求,则系统就会立即在物理内存中就会为该对象开辟空间,即使我们不对该对象做任何的使用,该对象也会静静的放在内存中。当我们真正要访问该对象时,只能通过唯一的途径访问。

        代码实现饿汉方式:

#include <iostream>#include <unistd.h>using namespace std;class Singleton{    Singleton()//该类不能在外被构造    {        cout<<"程序运行时就打印该信息"<<endl;    }    ~Singleton(){}//该类不能在外被析构    Singleton(const Singleton& )=delete;//该类不能在外被拷贝    Singleton& operator=(const Singleton& )=delete;//该类不能在外被赋值        static Singleton data;//该进程下唯一Singleton对象,即单例public:    static Singleton *GetInstance()//外部只能通过调用该函数拿到data    {        return &data;    }};Singleton Singleton::data;//data的初始化int main(){        return 0;}

        运行结果:

        从结果可以发现,代码中仅仅只是定义了一个类,还未手动对该类进行实例化,程序开始执行时系统就自动实例化了一个Singleton的data对象,因为Singleton类的构造函数被执行,说明构造了一个对象。

        对上述代码的逻辑可以理解为:定义Singleton类就是提出需求,而程序一启动就执行构造函数说明提出需求后就立即得到了一个对象。


        并且在该进程下只存在唯一的Singleton类型对象,并且访问该唯一对象的途径只能是调用静态成员函数GetInstance,并且当调用GetInstance静态函数说明“真正用到了该对象”,测试代码如下:

#include <iostream>#include <unistd.h>using namespace std;class Singleton{    Singleton()//该类不能在外被构造    {        cout<<"程序运行时就打印该信息"<<endl;    }    ~Singleton(){}//该类不能在外被析构    Singleton(const Singleton& )=delete;//该类不能在外被拷贝    Singleton& operator=(const Singleton& )=delete;//该类不能在外被赋值        static Singleton data;//该进程下唯一Singleton对象,即单例public:    static Singleton *GetInstance()//外部只能通过调用该函数拿到data    {        return &data;    }};Singleton Singleton::data;//data的初始化int main(){    //以下两种实例化方式都无法实例化出对象    // Singleton s;    // Singleton s1(s);    cout<<Singleton::GetInstance()<<endl;    return 0;}

        运行结果:

2、懒汉方式实现单例

        懒汉方式指的是当提出需求后,系统不会立即实例化出对象,而是先实例化一个指针,当真正需要用到该对象时,系统才会主动的去申请一个对象,并让该指针指向这个对象。

        懒汉方式的代码如下:

#include <iostream>#include <unistd.h>using namespace std;class Singleton // 懒汉{    Singleton() // 该类不能在外被构造    {        cout << "程序运行时就打印该信息" << endl;    }    ~Singleton() {}                                   // 该类不能在外被析构    Singleton(const Singleton &) = delete;            // 该类不能在外被拷贝    Singleton &operator=(const Singleton &) = delete; // 该类不能在外被赋值    static Singleton *data;       // 使用Singleton*作为访问唯一对象的入口    static pthread_mutex_t lock_; // 保证线程安全public:    static Singleton *GetInstance() // 外部只能通过调用该函数拿到data指针    {        if (data == nullptr)        {            pthread_mutex_lock(&lock_);            if (data == nullptr)                data = new Singleton();            pthread_mutex_unlock(&lock_);        }        return data;    }};Singleton* Singleton::data = nullptr; // data的初始化pthread_mutex_t Singleton::lock_ = PTHREAD_MUTEX_INITIALIZER;//锁的初始化int main(){        return 0;}

        运行结果:

        可以发现,当程序一启动时,并没有直接构造一个Singleton类型的对象,因为没有调用Singleton的构造函数,只是简单的将data指针初始化为nullptr,只有当调用静态函数GetInstance时,系统才会实例化出对象,因为调用GetInstance表示要用到该对象,这时候就可以实例化对象了。


        调用函数GetInstance的代码如下:

#include <iostream>#include <unistd.h>using namespace std;class Singleton // 懒汉{    Singleton() // 该类不能在外被构造    {        cout << "程序运行时就打印该信息" << endl;    }    ~Singleton() {}                                   // 该类不能在外被析构    Singleton(const Singleton &) = delete;            // 该类不能在外被拷贝    Singleton &operator=(const Singleton &) = delete; // 该类不能在外被赋值    static Singleton *data;       // 使用Singleton*作为访问唯一对象的入口    static pthread_mutex_t lock_; // 保证线程安全public:    static Singleton *GetInstance() // 外部只能通过调用该函数拿到data指针    {        if (data == nullptr)        {            pthread_mutex_lock(&lock_);            if (data == nullptr)                data = new Singleton();            pthread_mutex_unlock(&lock_);        }        return data;    }};Singleton* Singleton::data = nullptr; // data的初始化pthread_mutex_t Singleton::lock_ = PTHREAD_MUTEX_INITIALIZER;//锁的初始化int main(){    //只有调用GetInstance时,才会开辟空间    cout << Singleton::GetInstance() << endl;    return 0;}

        运行结果:

        从结果可以看到,只要真正要用到该对象,系统才会实例化该对象,没有用到该对象前,系统只有一个指针做“准备就绪”的工作。 

3、单例模式的总结 

        不管是饿汉方式还是懒汉方式,基本实现都是依靠static修饰的成员变量,并且该静态变量要放在类内,因为单例模式下构造函数是在私有域中的, 静态变量只有在类内才可以调用该类的构造函数进行初始化,这也从侧面表示出静态成员变量的类型必须和当前类的类型是一样的。拿到该类的实例化对象的途径只有通过调用该类的静态成员函数去访问。

        一般懒汉方式用的最多,因为懒汉在局部上加快了速度,因为他改变的是花费时间的结构,比如要加载某个任务,若把整个任务都加载下来则需要很多时间,但是我们可以先加载任务的一小部分先用上,而无需等待整个任务都加载下来,等到真正使用该任务的时候在进行下载,可以将空间利用率最大化。

结语

        以上就是关于单例模式的讲解,单例模式下用的最多的是懒汉方式,因为他可以将内存的空间利用率最大化,无论是饿汉方式还是懒汉方式,本质上是利用了static静态变量在进程的唯一性。

        最后如果本文有遗漏或者有误的地方欢迎大家在评论区补充,谢谢大家!! 


点击全文阅读


本文链接:http://m.zhangshiyu.com/post/172065.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

最新文章

  • 祖母寿宴,侯府冒牌嫡女被打脸了(沈屿安秦秀婉)阅读 -
  • 《雕花锦年,昭都旧梦》(裴辞鹤昭都)完结版小说全文免费阅读_最新热门小说《雕花锦年,昭都旧梦》(裴辞鹤昭都) -
  • 郊区41号(许洛竹王云云)完整版免费阅读_最新全本小说郊区41号(许洛竹王云云) -
  • 负我情深几许(白诗茵陆司宴)完结版小说阅读_最热门小说排行榜负我情深几许白诗茵陆司宴 -
  • 九胞胎孕妇赖上我萱萱蓉蓉免费阅读全文_免费小说在线看九胞胎孕妇赖上我萱萱蓉蓉 -
  • 为保白月光,侯爷拿我抵了债(谢景安花田)小说完结版_完结版小说全文免费阅读为保白月光,侯爷拿我抵了债谢景安花田 -
  • 陆望程映川上官硕《我的阿爹是带攻略系统的替身》最新章节阅读_(我的阿爹是带攻略系统的替身)全章节免费在线阅读陆望程映川上官硕
  • 郑雅琴魏旭明免费阅读_郑雅琴魏旭明小说全文阅读笔趣阁
  • 头条热门小说《乔书意贺宴临(乔书意贺宴临)》乔书意贺宴临(全集完整小说大结局)全文阅读笔趣阁
  • 完结好看小说跨年夜,老婆初恋送儿子故意出车祸_沈月柔林瀚枫完结的小说免费阅读推荐
  • 热推《郑雅琴魏旭明》郑雅琴魏旭明~小说全文阅读~完本【已完结】笔趣阁
  • 《你的遗憾与我无关》宋怀川冯洛洛无弹窗小说免费阅读_免费小说大全《你的遗憾与我无关》宋怀川冯洛洛 -

    关于我们 | 我要投稿 | 免责申明

    Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1