《C++新经典设计模式》之第7章 单例模式

《C++新经典设计模式》之第7章 单例模式

        • 单例模式.cpp

单例模式.cpp
#include <iostream>
#include <memory>
#include <mutex>
#include <vector>
#include <atomic>
using namespace std;// 懒汉式,未释放
namespace ns1
{class GameConfig{static GameConfig *m_instance1; // 指向本类对象的指针// static mutex m_mutex;// static shared_ptr<GameConfig> GameConfig::m_instance3;private:GameConfig(){};GameConfig(const GameConfig &tmpobj);GameConfig &operator=(const GameConfig &tmpobj);~GameConfig(){};public:static GameConfig *getInstance1(){// lock_guard<mutex> l(m_mutex);if (m_instance1 == nullptr)m_instance1 = new GameConfig();return m_instance1;}static GameConfig *getInstance2(){static GameConfig m_instance2;return &m_instance2;}/*static shared_ptr<GameConfig> getInstance3(){if (m_instance3 == nullptr)m_instance3 = make_shared<GameConfig>();return m_instance3;}*//*static shared_ptr<GameConfig> getInstance4(){static shared_ptr<GameConfig> m_instance4(new GameConfig());return m_instance4;}*/};GameConfig *GameConfig::m_instance1 = nullptr;// mutex GameConfig::m_mutex;// shared_ptr<GameConfig> GameConfig::m_instance3 = nullptr;
}// 懒汉式,嵌套类释放
namespace ns2
{class GameConfig{static GameConfig *m_instance;// static mutex m_mutex;private:GameConfig(){};GameConfig(const GameConfig &tmpobj);GameConfig &operator=(const GameConfig &tmpobj);~GameConfig(){};public:static GameConfig *getInstance(){// lock_guard<mutex> gcguard(m_mutex);if (m_instance == nullptr){// 这里再加锁// lock_guard<mutex> gcguard(m_mutex);// if (m_instance == nullptr)//{m_instance = new GameConfig();static Garbo garboobj;//}}return m_instance;}public:static void freeInstance() // 要手工调用才能释放内存{if (GameConfig::m_instance != nullptr){delete GameConfig::m_instance;GameConfig::m_instance = nullptr;}}private:class Garbo // 手工释放单件类对象引入的GameConfig类中的嵌套类(垃圾回收){public:~Garbo() { GameConfig::freeInstance(); }};};GameConfig *GameConfig::m_instance = nullptr;// mutex GameConfig::m_mutex;
}// 懒汉式,加锁+嵌套类释放
namespace ns3
{class GameConfig{static atomic<GameConfig *> m_instance;static mutex m_mutex;private:GameConfig(){};GameConfig(const GameConfig &tmpobj);GameConfig &operator=(const GameConfig &tmpobj);~GameConfig(){};public:static GameConfig *getInstance(){GameConfig *tmp = m_instance.load(memory_order_relaxed);atomic_thread_fence(memory_order_acquire);if (tmp == nullptr){lock_guard<mutex> lock(m_mutex);tmp = m_instance.load(memory_order_relaxed);if (tmp == nullptr){tmp = new GameConfig();static Garbo garboobj;atomic_thread_fence(memory_order_release);m_instance.store(tmp, memory_order_relaxed);}}return tmp;}private:class Garbo // 手工释放单件类对象引入的GameConfig类中的嵌套类(垃圾回收){public:~Garbo(){if (GameConfig::m_instance != nullptr){delete GameConfig::m_instance;GameConfig::m_instance = nullptr;}}};};atomic<GameConfig *> GameConfig::m_instance;mutex GameConfig::m_mutex;
}// 饿汉式,嵌套类释放
namespace ns4
{class GameConfig{static GameConfig *m_instance;private:GameConfig(){};GameConfig(const GameConfig &tmpobj);GameConfig &operator=(const GameConfig &tmpobj);~GameConfig(){};public:static GameConfig *getInstance() { return m_instance; }private:class Garbo // 手工释放单件类对象引入的GameConfig类中的嵌套类(垃圾回收){public:~Garbo(){if (GameConfig::m_instance != nullptr){delete GameConfig::m_instance;GameConfig::m_instance = nullptr;}}};private:static Garbo garboobj;};GameConfig *GameConfig::m_instance = new GameConfig(); // 趁静态成员变量定义的时机直接初始化是被允许的,即便GameConfig构造函数用private修饰GameConfig::Garbo GameConfig::garboobj;
}// 饿汉式
namespace ns5
{class GameConfig2{static GameConfig2 m_instance;private:GameConfig2(){};GameConfig2(const GameConfig2 &tmpobj);GameConfig2 &operator=(const GameConfig2 &tmpobj);~GameConfig2(){};public:static GameConfig2 *getInstance() { return &m_instance; }};GameConfig2 GameConfig2::m_instance;class GameConfig{private:GameConfig(){};GameConfig(const GameConfig &tmpobj);GameConfig &operator=(const GameConfig &tmpobj);~GameConfig(){};public:static GameConfig &getInstance(){static GameConfig instance;return instance;}};int myfunc(){static int stcs = 100; // 不需要调用myfunc函数,stcs就已经等于100了stcs += 180;return stcs;}
}// 饿汉式
namespace ns111
{// 饿汉模式class CSingleton{CSingleton() {}~CSingleton() {}public:static CSingleton *GetInstance(){static CSingleton instance;return &instance;}};// 饿汉模式2class CSingleton2{static CSingleton2 *instance;CSingleton2() {}~CSingleton2() {}public:static CSingleton2 *GetInstance(){return instance;}};CSingleton2 *CSingleton2::instance = new CSingleton2();// 饿汉模式3,智能指针class CSingleton3{static shared_ptr<CSingleton3> instance;CSingleton3() {}~CSingleton3() {}public:static shared_ptr<CSingleton3> GetInstance(){return instance;}static void destoryInstance(CSingleton3 *x){delete x;}};shared_ptr<CSingleton3> CSingleton3::instance(new CSingleton3(), destoryInstance);// 饿汉模式4,静态嵌套类class CSingleton4{static CSingleton4 *instance;class Deleter{public:~Deleter(){if (instance != nullptr){delete instance;instance = nullptr;}}};static Deleter m_deleter;CSingleton4() {}~CSingleton4() {}public:static CSingleton4 *GetInstance() { return instance; }};CSingleton4 *CSingleton4::instance = new CSingleton4();CSingleton4::Deleter CSingleton4::m_deleter;
}namespace ns222
{// 懒汉模式1class Singleton1{static Singleton1 *instance;Singleton1() {}~Singleton1() {}public:static Singleton1 *getInstance(){if (instance == nullptr)instance = new Singleton1();return instance;}};Singleton1 *Singleton1::instance = nullptr;// 懒汉模式2class Singleton2{static Singleton2 *m_instance;static mutex mtx;Singleton2() {}~Singleton2() {}public:Singleton2(const Singleton2 &) = delete;Singleton2 &operator=(const Singleton2 &) = delete;public:static Singleton2 *getInstance(){if (NULL == m_instance){unique_lock<mutex> lock(mtx);if (NULL == m_instance)m_instance = new Singleton2();}return m_instance;}};Singleton2 *Singleton2::m_instance = nullptr;mutex Singleton2::mtx;// 懒汉模式3,智能指针class Singleton3{static shared_ptr<Singleton3> m_instance;static mutex mtx;Singleton3() {}~Singleton3() {}public:Singleton3(const Singleton3 &) = delete;Singleton3(Singleton3 &&) = delete;Singleton3 &operator=(const Singleton3 &) = delete;Singleton3 &operator=(Singleton3 &&) = delete;public:static void destoryInstance(Singleton3 *x) { delete x; }static shared_ptr<Singleton3> getInstance(){if (NULL == m_instance){lock_guard<mutex> lock(mtx);if (NULL == m_instance)m_instance.reset(new Singleton3(), destoryInstance);}return m_instance;}};shared_ptr<Singleton3> Singleton3::m_instance = nullptr;mutex Singleton3::mtx;// 懒汉模式4,静态嵌套类class Singleton4{static Singleton4 *m_instance;static mutex mtx;Singleton4() {}~Singleton4() {}class Deleter{public:~Deleter(){if (m_instance != nullptr){delete m_instance;m_instance = nullptr;}}};static Deleter m_deleter;public:Singleton4(const Singleton4 &) = delete;Singleton4 &operator=(const Singleton4 &) = delete;public:static Singleton4 *getInstance(){if (NULL == m_instance){lock_guard<mutex> lock(mtx);if (NULL == m_instance)m_instance = new Singleton4();}return m_instance;}};Singleton4 *Singleton4::m_instance = nullptr;mutex Singleton4::mtx;Singleton4::Deleter Singleton4::m_deleter;// 懒汉模式5class Singleton5{static atomic<Singleton5 *> m_instance;static mutex mtx;Singleton5() {}~Singleton5() {}public:Singleton5(const Singleton5 &) = delete;Singleton5 &operator=(const Singleton5 &) = delete;public:static Singleton5 *getInstance(){Singleton5 *p = m_instance;if (p == nullptr){lock_guard<mutex> ll{mtx};if ((p = m_instance) == nullptr)m_instance = p = new Singleton5();}return m_instance;}};atomic<Singleton5 *> Singleton5::m_instance{nullptr};mutex Singleton5::mtx;
}//  加锁的懒汉式
namespace ns2_1
{class SingleInstance{private:static SingleInstance *m_SingleInstance;static mutex m_Mutex;private:SingleInstance() {}~SingleInstance() {}SingleInstance(const SingleInstance &signal) {}const SingleInstance &operator=(const SingleInstance &signal) {}public:static SingleInstance *GetInstance(){if (m_SingleInstance == nullptr){unique_lock<mutex> lock(m_Mutex); // 加锁if (m_SingleInstance == nullptr){volatile auto temp = new (nothrow) SingleInstance();m_SingleInstance = temp;}}return m_SingleInstance;}static void deleteInstance(){unique_lock<mutex> lock(m_Mutex); // 加锁if (m_SingleInstance){delete m_SingleInstance;m_SingleInstance = nullptr;}}void Print() { cout << "实例内存地址: " << this << endl; }};SingleInstance *SingleInstance::m_SingleInstance = nullptr;mutex SingleInstance::m_Mutex;
}namespace ns2_2
{class Singleton{static shared_ptr<Singleton> singleton;static mutex singletonMutex;public:static shared_ptr<Singleton> getSingleton(){if (singleton == nullptr){unique_lock<mutex> lock(singletonMutex);if (singleton == nullptr)singleton.reset(new Singleton());}return singleton;}void print() { cout << "Hello World." << endl; }~Singleton() { cout << __PRETTY_FUNCTION__ << endl; }private:Singleton() { cout << __PRETTY_FUNCTION__ << endl; }};shared_ptr<Singleton> Singleton::singleton = nullptr;mutex Singleton::singletonMutex;
}namespace ns2_3
{class Singleton{static shared_ptr<Singleton> singleton;static once_flag singletonFlag;private:Singleton() {}// public://     ~Singleton() {}public:void print() { cout << "Hello World." << endl; }public:static shared_ptr<Singleton> getSingleton(){call_once(singletonFlag, [&]{ singleton.reset(new Singleton()); });return singleton;}};shared_ptr<Singleton> Singleton::singleton = nullptr;once_flag Singleton::singletonFlag;
}// 模板实现
namespace ns2_4
{template <typename T>shared_ptr<T> getInstance(){static shared_ptr<T> instance(new T());return instance;}class TestClass{public:TestClass() { cout << "TestClass constructed!" << endl; }void print() { cout << "TestClass Address: " << this << endl; }};
}namespace ns22222
{class Single{private:Single() {}//~Single() {}static shared_ptr<Single> instance;Single(const Single &);Single &operator=(const Single &);public:static Single &getInstance(){if (instance == nullptr)instance.reset(new Single());return *instance;}};shared_ptr<Single> Single::instance = nullptr;class SingleObject{static shared_ptr<SingleObject> instance;SingleObject() {}// public://~SingleObject() {}public:static shared_ptr<SingleObject> getInstance() { return instance; }void showMessage() { cout << "Hello World!" << endl; }};shared_ptr<SingleObject> SingleObject::instance(new SingleObject());
}int main()
{
#if 0using namespace ns1;GameConfig *g_gc1 = GameConfig::getInstance1();GameConfig *g_gc2 = GameConfig::getInstance2();
#endif#if 0using namespace ns2;GameConfig *g_gc = GameConfig::getInstance();// g_gc->freeInstance(); // 手工释放内存演示
#endif#if 0using namespace ns3;GameConfig *g_gc = GameConfig::getInstance();GameConfig *g_gc2 = GameConfig::getInstance();
#endif#if 0using namespace ns4;GameConfig *g_gc = GameConfig::getInstance();GameConfig *g_gc2 = GameConfig::getInstance();
#endif#if 0using namespace ns5;GameConfig2 *g_gc = GameConfig2::getInstance();GameConfig &g_gc2 = GameConfig::getInstance();myfunc();
#endif#if 0using namespace ns2_1;SingleInstance *s = SingleInstance::GetInstance();s->Print();SingleInstance::GetInstance()->Print();SingleInstance::deleteInstance();
#endif#if 1using namespace ns2_4;getInstance<TestClass>()->print();getInstance<TestClass>()->print();auto instance1 = getInstance<vector<int>>();auto instance2 = getInstance<vector<int>>();instance1->push_back(1);instance2->push_back(2);cout << "instance1->size(): " << instance1->size() << endl;cout << "instance2->size(): " << instance2->size() << endl;cout << "instance1 address: " << instance1.get() << endl;cout << "instance2 address: " << instance2.get() << endl;
#endifcout << "Over!\n";return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/215581.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

手动搭建koa+ts项目框架(日志篇)

文章目录 前言一、安装koa-logger二、引入koa-logger并使用总结如有启发&#xff0c;可点赞收藏哟~ 前言 本文基于手动搭建koats项目框架&#xff08;路由篇&#xff09;新增日志记录 一、安装koa-logger npm i -S koa-onerror and npm i -D types/koa-logger二、引入koa-lo…

【每日一题】【12.11】1631.最小体力消耗路径

&#x1f525;博客主页&#xff1a; A_SHOWY&#x1f3a5;系列专栏&#xff1a;力扣刷题总结录 数据结构 云计算 数字图像处理 1631. 最小体力消耗路径https://leetcode.cn/problems/path-with-minimum-effort/这道题目的核心思路是&#xff1a;使用了二分查找和BFS &a…

PHP基础(2)

目录 一、PHP 数据类型 二、PHP 字符操作函数 strlen() str_word_count() strrev() strpos() str_replace() 一、PHP 数据类型 PHP 有八种基本数据类型和两种复合数据类型&#xff1a; 整型&#xff08;int&#xff09;&#xff1a;表示整数&#xff0c;可以是正数或负数&am…

线程Thread源代码思想学习1

1.启动线程代码 public class MultiThreadExample {public static void main(String[] args) {// 创建两个线程对象Thread thread1 new Thread(new Task());Thread thread2 new Thread(new Task());// 启动线程thread1.start();thread2.start();} }class Task implements Ru…

EXPLAIN 执行计划

有了慢查询语句后&#xff0c;就要对语句进行分析。一条查询语句在经过 MySQL 查询优化器的各种基于成本和规则的优化会后生成一个所谓的执行计划&#xff0c;这个执行计划展示了接下来具体执行查询的方式&#xff0c;比如多表连接的顺序是什么&#xff0c;对于每个表采用什么访…

记录 DevEco 开发 HarmonyOS 应用开发问题记录 【持续更新】

HarmonyOS 应用开发问题记录 HarmonyOS 应用开发问题记录一、预览器无法成功运行?如何定位预览器无法编译问题? 开发遇到的问题 HarmonyOS 应用开发问题记录 一、预览器无法成功运行? 大家看到这个是不是很头疼? 网上能看到许多方案,基本都是关闭一个配置 但是他们并…

InitializingBean初始化--Spring容器管理

目录 InitializingBean--自动执行一些初始化操作spring初始化bean有两种方式&#xff1a;1.优点2.缺点2.PostConstruct 注解2.举例使用InitializingBean接口 和PostConstruct3.初始化交给容器管理4.与main入口函数有什么区别5.在 Spring 中&#xff0c;有两种主要的初始化 bean…

【Java SE】带你识别什么叫做异常!!!

&#x1f339;&#x1f339;&#x1f339;个人主页&#x1f339;&#x1f339;&#x1f339; 【&#x1f339;&#x1f339;&#x1f339;Java SE 专栏&#x1f339;&#x1f339;&#x1f339;】 &#x1f339;&#x1f339;&#x1f339;上一篇文章&#xff1a;【Java SE】带…

Android获取Wifi网关

公司有这样一个应用场景&#xff1a;有一台球机设备&#xff0c;是Android系统的&#xff0c;它不像手机&#xff0c;它没有触摸屏幕&#xff0c;所以我们对球机的操作很不方便&#xff0c;于是我们搞这样一个设置&#xff1a;点击球机电源键5次分享出一个热点&#xff0c;然后…

【JVM从入门到实战】(一) 字节码文件

一、什么是JVM JVM 全称是 Java Virtual Machine&#xff0c;中文译名 Java虚拟机。 JVM 本质上是一个运行在计算机上的程序&#xff0c;他的职责是运行Java字节码文件。 二、JVM的功能 解释和运行 对字节码文件中的指令&#xff0c;实时的解释成机器码&#xff0c;让计算机…

C++类模板不是一开始就创建的,而是调用时生成

类模板中的成员函数和普通类中成员函数创建时机有区别的&#xff1a; 普通类中的成员函数一开始就可以创建模板类中的成员函数调用的时候才可以创建 总结;类模板中的成员函数并不是一开始就创建的&#xff0c;再调用时才去创建 #include<iostream> using namespace st…

微信小程序:模态框(弹窗)的实现

效果 wxml <!--新增&#xff08;点击按钮&#xff09;--> <image classimg src"{{add}}" bindtapadd_mode></image> <!-- 弹窗 --> <view class"modal" wx:if"{{showModal}}"><view class"modal-conten…

Vue中$props、$attrs和$listeners的使用详解

文章目录 透传属性如何禁止“透传属性和事件”多根节点设置透传访问“透传属性和事件” $props、$attrs和$listeners的使用详解 透传属性 透传属性和事件并没有在子组件中用props和emits声明透传属性和事件最常见的如click和class、id、style当子组件只有一个根元素时&#xf…

jOOQ指南中使用的数据库

jOOQ指南中使用的数据库 本指南中使用的数据库将在本节中进行总结和创建 使用Oracle方言来创建 # 创建语言 CREATE TABLE language (id NUMBER(7) NOT NULL PRIMARY KEY,cd CHAR(2) NOT NULL,description VARCHAR2(50) );# 创建作者 CREATE TABLE author (id NUMBER(7) NOT …

mysql:需要准确存储的带小数的数据,要使用DECIMAL类型

需要准确存储的带小数的数据&#xff0c;要使用DECIMAL&#xff0c;特别是涉及金钱类的业务。而不要使用FLOAT或者DOUBLE。 因为DECIMAL是准确值&#xff0c;不会损失精度。 而FLOAT或者DOUBLE是近似值&#xff0c;会损失精度。 https://dev.mysql.com/doc/refman/8.2/en/fixe…

差生文具多系列之最好看的编程字体

&#x1f4e2; 声明&#xff1a; &#x1f344; 大家好&#xff0c;我是风筝 &#x1f30d; 作者主页&#xff1a;【古时的风筝CSDN主页】。 ⚠️ 本文目的为个人学习记录及知识分享。如果有什么不正确、不严谨的地方请及时指正&#xff0c;不胜感激。 直达博主&#xff1a;「…

数据结构 | Floyd

参考博文&#xff1a; floyd算法 弗洛伊德算法 多源最短路径算法_弗洛伊德算法例题-CSDN博客

【文心一言】AI试用写代码体会:delphi、php

一、前言&#xff1a; 二、让【文心一言】编写一个函数 1. Delphi 语言&#xff08;文心一言的回复&#xff09; 2. php 语言&#xff08;文心一言回复&#xff09; 三、总结 一、前言&#xff1a; 众所周知&#xff0c;chatGPT是可以自动编写程序的&#xff0c;甚至可以运…

MyBatis缓存机制流程分析

前言 在进行分析之前&#xff0c;建议快速浏览之前写的理解MyBatis原理、思想&#xff0c;这样更容易阅读、理解本篇内容。 验证一级缓存 MyBatis的缓存有两级&#xff0c;一级缓存默认开启&#xff0c;二级缓存需要手动开启。 重复读取跑缓存 可以看到&#xff0c;第二次…

OpenAI承认GPT-4变懒,即将发布修复方案提升性能

目录 1OpenAI承认GPT-4变懒&#xff0c;即将发布修复方案提升性能 2一文秒懂人工智能全球近况 1OpenAI承认GPT-4变懒&#xff0c;即将发布修复方案提升性能 **划重点:** 1. &#x1f92f; 用户反馈:GPT-4使用者抱怨OpenAI破坏了体验&#xff0c;称模型几乎“害怕”提供答案。…