热门标签 | 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 中提供的不可变集合(如 `Collections.unmodifiableXXX`)和同步集合(如 `Collections.synchronizedXXX`)的实现原理及使用方法,帮助开发者更好地理解和应用这些工具。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • JavaScript 基础语法指南
    本文详细介绍了 JavaScript 的基础语法,包括变量、数据类型、运算符、语句和函数等内容,旨在为初学者提供全面的入门指导。 ... [详细]
  • 本文详细介绍了优化DB2数据库性能的多种方法,涵盖统计信息更新、缓冲池调整、日志缓冲区配置、应用程序堆大小设置、排序堆参数调整、代理程序管理、锁机制优化、活动应用程序限制、页清除程序配置、I/O服务器数量设定以及编入组提交数调整等方面。通过这些技术手段,可以显著提升数据库的运行效率和响应速度。 ... [详细]
  • 深入解析Spring启动过程
    本文详细介绍了Spring框架的启动流程,帮助开发者理解其内部机制。通过具体示例和代码片段,解释了Bean定义、工厂类、读取器以及条件评估等关键概念,使读者能够更全面地掌握Spring的初始化过程。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • ListView简单使用
    先上效果:主要实现了Listview的绑定和点击事件。项目资源结构如下:先创建一个动物类,用来装载数据:Animal类如下:packagecom.example.simplelis ... [详细]
  • 本文详细探讨了HTML表单中GET和POST请求的区别,包括它们的工作原理、数据传输方式、安全性及适用场景。同时,通过实例展示了如何在Servlet中处理这两种请求。 ... [详细]
  • 本文探讨了在C++中如何有效地清空输入缓冲区,确保程序只处理最近的输入并丢弃多余的输入。我们将介绍一种不阻塞的方法,并提供一个具体的实现方案。 ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 本文介绍了如何使用JavaScript的Fetch API与Express服务器进行交互,涵盖了GET、POST、PUT和DELETE请求的实现,并展示了如何处理JSON响应。 ... [详细]
  • 深入解析Java虚拟机(JVM)架构与原理
    本文旨在为读者提供对Java虚拟机(JVM)的全面理解,涵盖其主要组成部分、工作原理及其在不同平台上的实现。通过详细探讨JVM的结构和内部机制,帮助开发者更好地掌握Java编程的核心技术。 ... [详细]
  • 在尝试使用C# Windows Forms客户端通过SignalR连接到ASP.NET服务器时,遇到了内部服务器错误(500)。本文将详细探讨问题的原因及解决方案。 ... [详细]
  • 本文探讨了如何利用HTML5和JavaScript在浏览器中进行本地文件的读取和写入操作,并介绍了获取本地文件路径的方法。HTML5提供了一系列API,使得这些操作变得更加简便和安全。 ... [详细]
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社区 版权所有