设计模式——单例模式(面试手撕顶呱呱)

简介: 设计模式——单例模式(面试手撕顶呱呱)

一、单例模式


1、单例模式定义:保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。

2、单例模式实现步骤:

(1)构造函数私有化;

(2)增加静态私有的当前类的指针变量;

(3)提供静态对外接口,可以让用户获得单例对象;


二、懒汉模式


1、方式一 (线程不安全)

//-----------------懒汉模式代码实例(线程不安全)---------------// 
//-------------------静态指针 + 用到时初始化------------------//  
class Singleton_lazy
{
public:
  static Singleton_lazy* getInstance()
  {
    if (pSingle_lazy == nullptr)
    {
      pSingle_lazy = new Singleton_lazy();
    }
    return pSingle_lazy;
  }
private:
  Singleton_lazy() { cout << "我是懒汉,线程安全!" << endl; };
  ~Singleton_lazy() {};
  static Singleton_lazy* pSingle_lazy;
};
Singleton_lazy* Singleton_lazy::pSingle_lazy = nullptr;

说明:


       懒汉模式在第一次用到类实例的时候才会被实例化!


       在单线程中,这样的写法是可以正确使用的,但是在多线程中就不行了,该方法是线程不安全的;


(1)假如线程A和线程B, 这两个线程要访问getInstance函数,线程A进入getInstance函数,并检测 if 条件,由于是第一次进入,value为空,if条件成立,准备创建对象实例。


(2)但是,线程A有可能被OS的调度器中断而挂起睡眠,而将控制权交给线程B。


(3) 线程B同样来到if条件,发现value还是为NULL,因为线程A还没来得及构造它就已经被中断 了。此时假设线程B完成了对象的创建,并顺利的返回。


(4)之后线程A被唤醒,继续执行new再次创建对象,这样一来,两个线程就构建两个对象实例, 这就破坏了唯一性。 另外,还存在内存泄漏的问题,new出来的东西始终没有释放;下面是一种饿汉式的一种改进。


2、方式二(线程安全)

//-----------------懒汉模式代码实例(线程安全)---------------// 
//-------------------静态指针 + 用到时初始化------------------//  
class Singleton_lazy
{
public:
  static Singleton_lazy* getInstance()
  {
    if (pSingle_lazy == nullptr)
    {
      pSingle_lazy = new Singleton_lazy();
    }
    return pSingle_lazy;
  }
private:
  class CGarbo
  {
  public:
    ~CGarbo()
    {
      if (Singleton_lazy::pSingle_lazy)
      {
        delete Singleton_lazy::pSingle_lazy;
      }
    }
  };
  static CGarbo Garbo;
  Singleton_lazy() { cout << "我是懒汉,线程安全!" << endl; };
  ~Singleton_lazy() {};
  static Singleton_lazy* pSingle_lazy;
};
Singleton_lazy* Singleton_lazy::pSingle_lazy = nullptr;


三、饿汉模式


//---------------饿汉模式代码实例(线程不安全)---------------//  
//-------------------静态指针 + 定义时初始化------------------//  
class Singleton_hungry
{
public:
  static Singleton_hungry* getInstance()
  {
    return pSingle_hungry;
  }
private:
  Singleton_hungry() { cout << "我是饿汉,线程安全!" << endl; };
  ~Singleton_hungry() {};
  static Singleton_hungry* pSingle_hungry;
};
Singleton_hungry* Singleton_hungry::pSingle_hungry = new Singleton_hungry;

说明:饿汉模式在定义的时候就进行实例化。因为main函数执行之前,全局作用域的类成员静态变量 m_Instance已经初始化,故没有多线程的问题。


四、测试


int main()
{
  cout << "程序开始运行!" << endl;
  Singleton_lazy::getInstance();    //懒汉模式只有在使用的时候才会被实例化
  Singleton_hungry::getInstance();  //饿汉模式在定义的时候被实例化
  return 0;
}

fe5bb0684b044b43ac8f34685e5167e1.png

相关文章
|
2月前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
31 2
|
14天前
|
设计模式 存储 前端开发
前端必须掌握的设计模式——单例模式
单例模式是一种简单的创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。适用于窗口对象、登录弹窗等场景,优点包括易于维护、访问和低消耗,但也有安全隐患、可能形成巨石对象及扩展性差等缺点。文中展示了JavaScript和TypeScript的实现方法。
|
19天前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
21 2
|
1月前
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
38 4
|
1月前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
25天前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
1月前
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入理解与应用
【10月更文挑战第22天】 在软件开发中,设计模式是解决特定问题的通用解决方案。本文将通过通俗易懂的语言和实例,深入探讨PHP中单例模式的概念、实现方法及其在实际开发中的应用,帮助读者更好地理解和运用这一重要的设计模式。
21 1
|
1月前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
27 0
|
2月前
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入解析与实践
在PHP开发中,设计模式是提高代码可维护性、扩展性和复用性的关键技术之一。本文将通过探讨单例模式,一种最常用的设计模式,来揭示其在PHP中的应用及优势。单例模式确保一个类仅有一个实例,并提供一个全局访问点。通过实际案例,我们将展示如何在PHP项目中有效实现单例模式,以及如何利用这一模式优化资源配置和管理。无论是PHP初学者还是经验丰富的开发者,都能从本文中获得有价值的见解和技巧,进而提升自己的编程实践。
|
2月前
|
设计模式 安全 Java
C# 一分钟浅谈:设计模式之单例模式
【10月更文挑战第9天】单例模式是软件开发中最常用的设计模式之一,旨在确保一个类只有一个实例,并提供一个全局访问点。本文介绍了单例模式的基本概念、实现方式(包括饿汉式、懒汉式和使用 `Lazy&lt;T&gt;` 的方法)、常见问题(如多线程和序列化问题)及其解决方案,并通过代码示例详细说明了这些内容。希望本文能帮助你在实际开发中更好地应用单例模式,提高代码质量和可维护性。
64 1