热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

单例模式及线程安全

文章目录多线程时的单例双重锁定饿汉式单例懒汉式单例使用内部类析构使用智能指针实现单例模式(Singleton):保证一个类仅有一个实例&#


文章目录

    • 多线程时的单例
    • 双重锁定
    • 饿汉式单例
    • 懒汉式单例
      • 使用内部类析构
      • 使用智能指针实现




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

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


构造方法让其private,堵死了外界利用new创建此类的实例的可能。


多线程时的单例

static Config * Config::instance()
{//加锁(多线程场景下)m_mutex.acquire();if (NULL == m_instance){m_instance = new Config();m_instance->init();}//解锁m_mutex.release();return m_instance;
}

双重锁定

static Config * instance(){if (NULL == m_instance){//加锁(多线程场景下)m_mutex.acquire();if (NULL == m_instance){Config * pInstance = new Config();pInstance->init();m_instance = pInstance;}//解锁m_mutex.release();}return m_instance;}

饿汉式单例


  • 饿汉式单例:即静态初始化的方式,是类一加载就实例化对象。所以要提前占用系统资源。就是不管你要不要都会直接创建一个对象。只能有一个static对象(线程安全)。

class Singleton
{
public:static Singleton* getInstance (){return &_instance;}
private:static Singleton _instance;Singleton() {};Singleton(const Singleton&) = delete;Singleton& operator=(const Singleton&) = delete;
};
Singleton Singleton::_instance;

懒汉式单例


  • 懒汉式单例:需要考虑多线程访问的安全性问题。
  • 没有公有构造方法,一个公有静态工厂方法,和一个静态实例变量构造函数私有

使用内部类析构

//一个较全的懒汉式单例模式
#include
#include
#include
using namespace std;
class Singleton
{
public:static Singleton *getInstance(){if (instance == nullptr){pthread_mutex_lock(& _mutex); //再进行 次判空有多个线程在第一次判//断空时进入了第一个if语句,此时如果不在进行判空在解锁后会有线程再次去进行new对象if (instance == nullptr)instance = new Singleton();pthread_mutex_unlock(& _mutex);}return instance;}
private:Singleton(){cout << "Singleton()" << endl;}Singleton(const Singleton &instance){cout <<"singleton (const Singleton&)" <<endl;}~Singleton() {pthread_mutex_destroy(&_mutex);cout << "~Singleton()" << endl;}class Release //自动析构instance类{public:~Release() //使用一个内部类&#xff0c;对instance进行delete{if(instance !&#61; nullptr){delete instance ;instance &#61; nullptr;}}};static Release rel;static Singleton *instance;static pthread_mutex_t _mutex;
};Singleton *Singleton::instance &#61; nullptr;
pthread_mutex_t Singleton::_mutex &#61; PTHREAD_MUTEX_INITIALIZER;
Singleton::Release Singleton::rel;int main() {Singleton *p1 &#61; Singleton::getInstance();Singleton *p2 &#61; Singleton::getInstance();Singleton *P3 &#61; p2;cout << p1 <<" "<< p2 << endl;if (p1 &#61;&#61; p2)cout << "两个对象是相同的实例" << endl;return 0;
}

输出

Singleton()
0x1041770 0x1041770
两个对象是相同的实例
~Singleton()

使用智能指针实现

问题&#xff1a;shared_ptr无法访问私有化的析构函数

#include
#include
#include
#include
#include
#include
using namespace std;
class Singleton
{
public:static std::shared_ptr<Singleton> getInstance(){if (instance &#61;&#61; nullptr){pthread_mutex_lock(& _mutex); //再进行 次判空有多个线程在第一次判//断空时进入了第一个if语句&#xff0c;此时如果不在进行判空在解锁后会有线程再次去进行new对象if (instance &#61;&#61; nullptr)instance.reset(new Singleton(),Singleton::DestroyInstance);pthread_mutex_unlock(& _mutex);}return instance;}static void print(){cout<<m_count<<endl;}private:Singleton(){cout << "Singleton()" << endl;Sleep(1000);m_count &#43;&#43;;// 加sleep为了放大效果}~Singleton() {pthread_mutex_destroy(&_mutex);cout << "~Singleton()" << endl;}Singleton(const Singleton &instance){cout <<"singleton (const Singleton&)" <<endl;}static void DestroyInstance(Singleton *instance) //自定义一个释放实例的函数{delete instance; // 释放单例对象}static int m_count;static std::shared_ptr<Singleton> instance;static pthread_mutex_t _mutex;
};// 回调函数
void threadFunc(void *p){DWORD id &#61; GetCurrentThreadId(); // 获得线程idcout<<id<<endl;Singleton::getInstance()->print(); // 构造函数并获得实例&#xff0c;调用静态成员函数
}std::shared_ptr<Singleton> Singleton::instance &#61; nullptr;
pthread_mutex_t Singleton::_mutex &#61; PTHREAD_MUTEX_INITIALIZER;
int Singleton::m_count &#61; 0;int main() {int threadNum &#61; 3;HANDLE threadHdl[100];// 创建3个线程for(int i &#61; 0; i<threadNum; i&#43;&#43;){threadHdl[i] &#61; (HANDLE)_beginthread(threadFunc, 0, nullptr);}// 让主进程等待所有的线程结束后再退出for(int i &#61; 0; i<threadNum; i&#43;&#43;){WaitForSingleObject(threadHdl[i], INFINITE);}return 0;
}

推荐阅读
  • 问题描述现在,不管开发一个多大的系统(至少我现在的部门是这样的),都会带一个日志功能;在实际开发过程中 ... [详细]
  • 本文基于Java官方文档进行了适当修改,旨在介绍如何实现一个能够同时处理多个客户端请求的服务端程序。在前文中,我们探讨了单客户端访问的服务端实现,而本篇将深入讲解多客户端环境下的服务端设计与实现。 ... [详细]
  • 题目描述:Balala Power! 时间限制:4000/2000 MS (Java/Other) 内存限制:131072/131072 K (Java/Other)。题目背景及问题描述详见正文。 ... [详细]
  • 本文针对HDU 1042 N! 问题提供详细的解析和代码实现。题目要求计算给定整数N(0 ≤ N ≤ 10000)的阶乘N!。文章不仅提供了算法思路,还附上了C++语言的具体实现。 ... [详细]
  • 该问题描述了以不同价格购买三种类型的鸡(公鸡、母鸡和小鸡),使用100元恰好购买100只鸡的不同组合。具体而言,每只公鸡价值5元,每只母鸡价值3元,而每三只小鸡价值1元。问题是,如何用100元购买100只鸡,并找出所有可能的公鸡、母鸡和小鸡的组合。 ... [详细]
  • Java多线程售票案例分析
    本文通过一个售票系统的实例,深入探讨了Java中的多线程技术及其在资源共享和并发控制中的应用。售票过程涉及查询、收款、找零和出票等多个步骤,其中对总票数的管理尤为关键。 ... [详细]
  • 编程解析:CF989C 花朵之雾 (构造算法)
    本文深入探讨了CF989C '花朵之雾'问题的构造算法,提供了详细的解题思路和代码实现。 ... [详细]
  • 探讨了一个包含纯虚函数的C++代码片段,分析了其中的语法错误及逻辑问题,并提出了修正方案。 ... [详细]
  • 想把一组chara[4096]的数组拷贝到shortb[6][256]中,尝试过用循环移位的方式,还用中间变量shortc[2048]的方式。得出的结论:1.移位方式效率最低2. ... [详细]
  • Hanks博士是一位著名的生物技术专家,他的儿子Hankson对数学有着浓厚的兴趣。最近,Hankson遇到了一个有趣的数学问题,涉及求解特定条件下的正整数x,而不使用传统的辗转相除法。 ... [详细]
  • 网络流24题——试题库问题
    题目描述:假设一个试题库中有n道试题。每道试题都标明了所属类别。同一道题可能有多个类别属性。现要从题库中抽取m道题组成试卷。并要求试卷包含指定类型的试题。试设计一个满足要求的组卷算 ... [详细]
  • hlg_oj_1116_选美大赛这题是最长子序列,然后再求出路径就可以了。开始写的比较乱,用数组什么的,后来用了指针就好办了。现在把代码贴 ... [详细]
  • 在1995年,Simon Plouffe 发现了一种特殊的求和方法来表示某些常数。两年后,Bailey 和 Borwein 在他们的论文中发表了这一发现,这种方法被命名为 Bailey-Borwein-Plouffe (BBP) 公式。该问题要求计算圆周率 π 的第 n 个十六进制数字。 ... [详细]
  • 本文详细介绍了 `org.apache.tinkerpop.gremlin.structure.VertexProperty` 类中的 `key()` 方法,并提供了多个实际应用的代码示例。通过这些示例,读者可以更好地理解该方法在图数据库操作中的具体用途。 ... [详细]
  • 二维码的实现与应用
    本文介绍了二维码的基本概念、分类及其优缺点,并详细描述了如何使用Java编程语言结合第三方库(如ZXing和qrcode.jar)来实现二维码的生成与解析。 ... [详细]
author-avatar
李da寕
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有