【Singleton Pattern】设计模式之单例模式

简介: 【Singleton Pattern】设计模式之单例模式

1.单例模式定义


单例模式:确保一个类只能有一个实例,并且提供一个全局访问来访问这个唯一实例。单例模式中的要点

(1) 该类只能有一个实例;

(2) 该类必须自己创建这个实例;

(3) 该类必须自己向整个系统提供这个实例;


2.单例模式UML图


单例模式结构简单,它的UML图如下所示。仅包含一个类,即单例类。为了防止创建多个对象,其构造函数必须是私有的(private),这样外界就不能访问。另外,为了能提供一个全局访问点来访问此唯一实例,单例类中提供了一个公有方法getInstance()来返回唯一实例


65.png


3.单例模式入门实战


#ifndef __SINGLETON_H__
#define __SINGLETON_H__
#include <iostream>
using namespace std;
// 单例类
class Singleton{
public:
    static Singleton* getInstance(){  // 外界通过static方法getInstance()方法获取单例对象,满足要点3
        if(instance == nullptr){
            cout << "创建一个新的实例对象\n";
            instance = new Singleton();
        }
        return instance;
    }
private:
    Singleton(){}   // 构造函数是私有的,即单例模式中对象仅能在单例类的内部实例化,满足要点2
    static Singleton* instance;  // 实例对象instance是static,即全局的,客户端程序中若要实例化两个Singleton对象,但instance仅有一个,满足要点1
};
Singleton* Singleton::instance = nullptr;
// 客户端程序
int main(){
    Singleton* s1 = Singleton::getInstance();
    Singleton* s2 = Singleton::getInstance();
    return 0;
}
#endif // __SINGLETON_H__


64.png


4.多线程环境下的单例模式


上面的入门实战程序中,实现了基本的单例模式。请思考一下多线程环境下如何实现安全的单例模式?在多线程环境中,当两个甚至多个线程同时使用,同样存在创建了多个实例对象的隐患问题。下面代码是多线程环境下,非线程安全的示例:


#ifndef __SINGLETON_H__
#define __SINGLETON_H__
#include <iostream>
#include <process.h>
#include <windows.h>
#define THREAD_NUM  5
using namespace std;
// 单例类
class Singleton{
public:
    static Singleton* getInstance(){  // 外界通过static方法getInstance()方法获取单例对象,满足要点3
        if(instance == nullptr){
            cout << "创建一个新的实例对象\n";
            instance = new Singleton();
        }
        return instance;
    }
private:
    Singleton(){}   // 构造函数是私有的,即单例模式中对象仅能在单例类的内部实例化,满足要点2
    static Singleton* instance;  // 实例对象instance是static,即全局的,客户端程序中若要实例化两个Singleton对象,但instance仅有一个,满足要点1
};
Singleton* Singleton::instance = nullptr;
unsigned int __stdcall CallSingleton(void *pPM){
    Singleton* s = Singleton::getInstance();
    int nThreadNum = *(int*)pPM;
    Sleep(50);
    cout << "线程编号: " << nThreadNum << endl;
    return 0;
}
// 客户端程序
int main(){
    HANDLE handle[THREAD_NUM];
    // 线程编号
    int threadNum = 0;
    while (threadNum < THREAD_NUM){
        handle[threadNum] = (HANDLE)_beginthreadex(nullptr, 0, CallSingleton, &threadNum, 0, nullptr);
        // 等子线程接收到参数时,主线程可能会改变这个i值
        threadNum++;
    }
    // 保证子线程已全部运行结束
    WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE);
    return 0;
}
#endif // __SINGLETON_H__

上面的程序中一共创建5个线程,每个线程中都会试图去创建一个单例对象。理论上,最终只有第一个线程(第一个被系统调度的线程)才能打印出“创建一个新的实例对象。以下结果说明:上面的单例模式代码并不是线程安全的


63.png


如何做到线程安全?多线程同步与互斥有多种方法,下面使用互斥锁来实现。


#ifndef __SINGLETON_H__
#define __SINGLETON_H__
#include <iostream>
#include <mutex>
#include <process.h>
#include <windows.h>
#define THREAD_NUM 5
using namespace std;
// 单例类
class Singleton
{
public:
    static Singleton *getInstance()
    {
        if (instance == nullptr)
        {
            m_mutex.lock();  // 互斥锁
            if (instance == nullptr)
            {
                cout << "创建一个新的实例对象\n";
                instance = new Singleton();
            }
            m_mutex.unlock();
        }
        return instance;
    }
private:
    Singleton() {}
    static Singleton *instance;
    static mutex m_mutex;
};
Singleton *Singleton::instance = nullptr;
mutex Singleton::m_mutex;
unsigned int __stdcall CallSingleton(void *pPM){
    Singleton* s = Singleton::getInstance();
    int nThreadNum = *(int*)pPM;
    Sleep(50);
    cout << "线程编号: " << nThreadNum << endl;
    return 0;
}
// 客户端程序
int main(){
    HANDLE handle[THREAD_NUM];
    // 线程编号
    int threadNum = 0;
    while (threadNum < THREAD_NUM){
        handle[threadNum] = (HANDLE)_beginthreadex(nullptr, 0, CallSingleton, &threadNum, 0, nullptr);
        // 等子线程接收到参数时,主线程可能会改变这个i值
        threadNum++;
    }
    // 保证子线程已全部运行结束
    WaitForMultipleObjects(THREAD_NUM, handle, TRUE, INFINITE);
    return 0;
}
#endif // __SINGLETON_H__


62.png


5.单例模式总结



优点:

(1) 单例模式提供了严格的对唯一实例对象的创建和访问

(2) 单例模式的实现可以节省系统资源



缺点:

(1) 多线程下使用单例模式,需要考虑线程安全问题

(2) 单例模式没有抽象层,不好扩展

相关文章
|
2月前
|
设计模式 存储 SQL
PHP中的设计模式:单例模式的探索
在PHP开发中,单例模式是一种常用的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将通过一个简单的例子,逐步引导你理解如何在PHP中实现和利用单例模式,以及它在实际项目中的应用价值。
|
8天前
|
设计模式 存储 安全
PHP中的设计模式:单例模式深度解析
在软件开发的广袤天地中,设计模式如同璀璨星辰,指引着程序员们解决复杂问题的方向。其中,单例模式以其独特的魅力,确保了一个类仅有一个实例,并提供了一个访问它的全局访问点,成为众多项目中不可或缺的设计智慧。本文旨在深入探讨PHP中单例模式的实现方式、应用场景及背后的哲理,引导读者思考其在现代软件架构中的重要性与运用策略。
23 5
|
13天前
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
23 11
|
9天前
|
设计模式 存储 缓存
PHP中的设计模式:单例模式的深入解析
在PHP开发中,设计模式是提高代码可维护性、扩展性和重用性的关键技术之一。本文将深入探讨PHP中的单例模式,包括其定义、实现方式、应用场景以及优缺点。通过对单例模式的全面剖析,帮助开发者更好地理解和应用这一设计模式,从而编写出更加高效和优雅的PHP代码。
|
9天前
|
设计模式
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
这篇文章详细解释了工厂模式,包括简单工厂、工厂方法和抽象工厂三种类型。每种模式都通过代码示例展示了其应用场景和实现方法,并比较了它们之间的差异。简单工厂模式通过一个工厂类来创建各种产品;工厂方法模式通过定义一个创建对象的接口,由子类决定实例化哪个类;抽象工厂模式提供一个创建相关或依赖对象家族的接口,而不需要明确指定具体类。
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
|
9天前
|
设计模式 安全 Java
设计模式--单例模式Singleton
这篇文章详细介绍了单例模式Singleton的八种实现方式,包括饿汉式(静态常量和静态代码块)、懒汉式(线程不安全和线程安全的同步方法、同步代码块)、双重检查、静态内部类和枚举。每种方式都有详细的代码示例和优缺点说明,帮助理解单例模式的应用和选择适合的实现方法。
设计模式--单例模式Singleton
|
11天前
|
设计模式 SQL 安全
PHP中的设计模式:单例模式的深入探索与实践在PHP的编程实践中,设计模式是解决常见软件设计问题的最佳实践。单例模式作为设计模式中的一种,确保一个类只有一个实例,并提供全局访问点,广泛应用于配置管理、日志记录和测试框架等场景。本文将深入探讨单例模式的原理、实现方式及其在PHP中的应用,帮助开发者更好地理解和运用这一设计模式。
在PHP开发中,单例模式通过确保类仅有一个实例并提供一个全局访问点,有效管理和访问共享资源。本文详细介绍了单例模式的概念、PHP实现方式及应用场景,并通过具体代码示例展示如何在PHP中实现单例模式以及如何在实际项目中正确使用它来优化代码结构和性能。
|
13天前
|
设计模式 数据库连接 PHP
PHP中的设计模式:单例模式的深入解析与实践
在PHP开发中,设计模式是提高代码可维护性、扩展性和复用性的关键技术之一。本文将深入探讨单例模式——一种确保类只有一个实例,并提供该实例的全局访问点的设计模式。我们将从单例模式的基本概念入手,剖析其在PHP中的应用方式,并通过实际案例展示如何在不同场景下有效利用单例模式来优化应用架构。
|
2月前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
9天前
|
设计模式 Java
设计模式--适配器模式 Adapter Pattern
这篇文章介绍了适配器模式,包括其基本介绍、工作原理以及类适配器模式、对象适配器模式和接口适配器模式三种实现方式。