06.单例模式Singleton

1.初识单例模式

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

Singleton:负责创建Singleton类自己的唯一实例,并提供一个getInstance的方法,让外部来访问这个类的唯一实例。

1.1 C++实现

//GOF提供的参考
class Singleton{
public:
    static Singleton* Instance();
protected:
    Singleton();
private:
    static Singleton* m_instance;
};

//实现
Singleton* Singleton::m_instance = nullptr;
Singleton* Singleton::Instance() {
    if (m_instance == nullptr) {
        m_instance = new Singleton();
    }
    return m_instance;
}
  • 使用惰性(lazy)初始化,Instance返回值直到第一次访问时才创建和保存。
  • Singleton
    定义一个Instance操作,允许客户访问它的唯一实例。Instance是一个类操作(C++中的一个静态成员函数)。
    可能负责创建它自己的唯一实例。
  • 客户只能通过Singleton的Instance操作访问一个Singleton的实例。

singleton单例的github源码

  • 说明1.关于c++ 11 static线程安全
    static只保证其本身的变量多线程是安全的,如果是指针的话,并不能保证其指向的内容是多线程安全!这也是3.1 3.2 与 3.3之间的差别!
  • 说明2.关于多线程安全
    假设两个线程同时判定m_instance为nullptr,则两个线程都会创建一个实例,此时就会出现问题。
    因此在多线程环境里面,需要加上一个同步锁。
  • 说明3.加锁是个耗时操作
    因此加上一个判断,只在为nullptr时才加锁。

2.体会单例模式

2.1 场景问题——读取配置文件的内容

现在要读取配置文件的内容,该如何实现呢?

2.2 不用模式的解决方案

存在的问题:
在系统运行期间,系统中会存在很多个AppConfig的实例对象,这会严重浪费系统资源。
把上面的描述进一步抽象一下,问题就出来了:在一个系统运行期间,某个类只需要一个类实例就可以了,那么应该怎么实现呢?

2.3 使用模式的解决方案

3.理解单例模式

3.1 认识单例模式

1.单例模式的功能单例模式的功能是用来保证这个类在运行期间只会被创建一个类实例,并提供一个全局唯一访问这个类实例的访问点。

2.单例模式的范围是一个ClassLoader及其子ClassLoader的范围

3.单例模式的命名一般建议单例模式的方法命名为:getInstance() 。

单例模式的名称:单例、单件、单体等等,翻译的不同,都是指的同一个模式

3.2 懒汉式和饿汉式实现

3.2.1 Java懒汉式

3.2.2 Java饿汉式

3.2.3 C++单线程实现(懒汉式)

//h文件
#include <iostream>
#include <cstddef>

#ifndef SINGLETON_H
#define SINGLETON_H

class Singleton {
public:
  static Singleton* GetInstance();
  static void DestroyInstance();

private:
  Singleton() {
    std::cout << "create a singleton" << std::endl;
  }
  class SingletonDel {
  public:
    ~SingletonDel() {
      if (m_instance != nullptr) {
        delete m_instance;
        m_instance = nullptr;
      }
    }

  };
  static SingletonDel m_singleton_del;
  static Singleton *m_instance;
};

#endif
//cc文件
#include <cstddef>
#include <iostream>
#include "singleton.h"


Singleton* Singleton::m_instance = nullptr;
Singleton* Singleton::GetInstance() {
  if (m_instance == nullptr) {
    m_instance = new Singleton();
  }
  std::cout << "m_instance: " << m_instance << std::endl;
  return m_instance;
}

void Singleton::DestroyInstance() {
  if (m_instance != nullptr) {
    delete m_instance;
    m_instance = nullptr;
  }
}
#include <thread>
#include "singleton.h"


int main() {
  std::thread t1(Singleton::GetInstance);
  t1.detach();
  std::thread t2(Singleton::GetInstance);
  t2.detach();
  std::thread t3(Singleton::GetInstance);
  t3.detach();
  std::thread t4(Singleton::GetInstance);
  t4.detach();
  std::thread t5(Singleton::GetInstance);
  t5.detach();
  std::thread t6(Singleton::GetInstance);
  t6.detach();
  std::thread t7(Singleton::GetInstance);
  t7.detach();

  return 0;
}

3.2.4 C++加锁实现线程安全(线程安全的懒汉式)

#include <iostream>
#include <cstddef>

#ifndef THREAD_SAFE_SINGLETON_H
#define THREAD_SAFE_SINGLETON_H

class ThreadSafeSingleton {
public:
    static ThreadSafeSingleton* GetInstance();
    static void DestroyInstance();
private:
    ThreadSafeSingleton() {
      std::cout << "create a thread safe singleton! " << std::endl;
    }

    class SingletonDel{
    public:
      ~SingletonDel() {
        if (m_instance != nullptr) {
          delete m_instance;
          m_instance = nullptr;
        }
      }
    };
    
    static ThreadSafeSingleton m_singleton_del;
    static ThreadSafeSingleton *m_instance;
};

#endif
#include "ThreadSafeSingleton.h"
#include <cstddef>
#include <mutex>
#include <iostream>

namespace thread_safe_singleton_private {
  static std::mutex thread_mutex;
}

ThreadSafeSingleton* ThreadSafeSingleton::m_instance = nullptr;
ThreadSafeSingleton* ThreadSafeSingleton::GetInstance() {
  if (m_instance == nullptr) {
    thread_safe_singleton_private::thread_mutex.lock();
    if (m_instance == nullptr) {
      m_instance = new ThreadSafeSingleton();
    }
    thread_safe_singleton_private::thread_mutex.unlock();
  }
  std::cout << "m_instance: " << m_instance << std::endl;
  return m_instance;
}

void ThreadSafeSingleton::DestroyInstance() {
  if (m_instance != nullptr) {
    delete m_instance;
    m_instance = nullptr;
  }
}

  • 使用c++11的lock_guard
    如下这段代码有个问题:
    第三行代码:if (pInstance == nullptr)和第六行代码pInstance = new Widget();没有正确的同步,在某种情况下会出现new返回了地址赋值给pInstance变量而Widget此时还没有构造完全,当另一个线程随后运行到第三行时将不会进入if从而返回了不完全的实例对象给用户使用,造成了严重的错误。在C++11没有出来的时候,只能靠插入两个memory barrier来解决这个错误,但是C++11已经出现了好几年了,其中我认为最重要的是引进了memory model,从此C++11也能识别线程这个概念了!
Widget* Widget::pInstance{ nullptr };
Widget* Widget::Instance() {
    if (pInstance == nullptr) { // 1: first check
        lock_guard<mutex> lock{ mutW };
        if (pInstance == nullptr) { // 2: second check
            pInstance = new Widget(); 
        }
    } 
    return pInstance;
}
  • 正确版本
    C++11中的atomic类的默认memory_order_seq_cst保证了3、6行代码的正确同步.
atomic<Widget*> Widget::pInstance{ nullptr };
Widget* Widget::Instance() {
    if (pInstance == nullptr) { 
        lock_guard<mutex> lock{ mutW }; 
        if (pInstance == nullptr) { 
            pInstance = new Widget(); 
        }
    } 
    return pInstance;
}
  • 优化版本
    由于上面的atomic需要一些性能上的损失,因此我们可以写一个优化的版本:(p不是atomic?)
atomic<Widget*> Widget::pInstance{ nullptr };
Widget* Widget::Instance() {
    Widget* p = pInstance;
    if (p == nullptr) { 
        lock_guard<mutex> lock{ mutW }; 
        if ((p = pInstance) == nullptr) { 
            pInstance = p = new Widget(); 
        }
    } 
    return p;
}
  • call_once实现单例模式
1 static unique_ptr<widget> widget::instance;
2 static std::once_flag widget::create;
3 widget& widget::get_instance() {
4     std::call_once(create, [=]{ instance = make_unique<widget>(); });
5     return instance;
6 }

3.2.5 c++11静态变量初始化是线程安全的(饿汉式)

1 widget& widget::get_instance() {
2     static widget instance;
3     return instance;
4 }
#include <iostream>

#ifndef CPP11_SINGLETON_H
#define CPP11_SINGLETON_H

template <typename T>
class Cpp11Singleton {
public:
  static T* GetInstance() {
    static T instance;
    std::cout << "this: " << &instance << std::endl;
    return &instance;
  }
};


class TestCpp11Singleton {
  friend class Cpp11Singleton<TestCpp11Singleton>;
private:
  TestCpp11Singleton() {
    std::cout << "create a cpp11 singleton! " <<  this << std::endl;
  }
};

#endif

3.3 延迟加载的思想

什么是延迟加载呢?
通俗点说,就是一开始不要加载资源或者数据,一直等到马上就要使用这个资源或者数据了,躲不过去了才加载,所以也称Lazy Load,不是懒惰啊,是 “延迟加载”,这在实际开发中是一种很常见的思想,尽可能的节约资源。

3.4 缓存的思想

单例模式的懒汉式实现还体现了缓存的思想,缓存也是实际开发中非常常见的功能。
简单讲就是,如果某些资源或者数据会被频繁的使用,可以把这些数据缓存到内存里面,每次操作的时候,先到内存里面找,看有没有这些数据,如果有,那么就直接使用,如果没有那么就获取它,并设置到缓存中,下一次访问的时候就可以直接从内存中获取了。从而节省大量的时间,当然,缓存是一种典型的空间换时间的方案。

3.5 Java中缓存的基本实现

3.6 利用缓存来实现单例模式

3.7 单例模式的优缺点

  • 时间和空间:懒汉式是典型的时间换空间,饿汉式是典型的空间换时间 * * * 线程安全:
    1)不加同步的懒汉式是线程不安全的



    2)饿汉式是线程安全的,因为虚拟机保证了只会装载一次
    3)如何实现懒汉式的线程安全呢?加上synchronized即可
    4)双重检查加锁
    所谓双重检查加锁机制,指的是:并不是每次进入getInstance方法都需要同步,而是先不同步,进入方法过后,先检查实例是否存在,如果不存在才进入下面的同步块,这是第一重检查。进入同步块过后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例,这是第二重检查。这样一来,就只需要同步一次了,从而减少了多次在同步情况下进行判断所浪费的时间。
    双重检查加锁机制的实现会使用一个关键字volatile,它的意思是:被 volatile修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。
    注意:在Java1.4及以前版本中,很多JVM对于volatile关键字的实现有问题,会导致双重检查加锁的失败,因此本机制只能用在Java5及以上的版本。

优点:

  • 对唯一实例的受控访问。
  • 缩小名空间。
    单例模式是对全局变量的一种改进,避免了全局变量污染全局命名空间。
  • 允许对操作和表示的精化。
    Singleton类可以有子类。(此时构造函数应该定义为protected)
  • 允许可变数目的实例
  • 比类操作更灵活

3.8 在Java中一种更好的单例实现方式

Lazy initialization holder class模式,这个模式综合使用了Java的类级内部类和多线程缺省同步锁的知识,很巧妙的同时实现了延迟加载和线程安全。

3.9 单例和枚举

按照《高效Java 第二版》中的说法:单元素的枚举类型已经成为实现 Singleton的最佳方法。
为了理解这个观点,先来了解一点相关的枚举知识,这里只是强化和总结一下枚举的一些重要观点,更多基本的枚举的使用,请参看Java编程入门资料:
1)Java的枚举类型实质上是功能齐全的类,因此可以有自己的属性和方法
2)Java枚举类型的基本思想:通过公有的静态final域为每个枚举常量导出实例的类
3)从某个角度讲,枚举是单例的泛型化,本质上是单元素的枚举

用枚举来实现单例非常简单,只需要编写一个包含单个元素的枚举类型即可。

4.思考单例模式

4.1 单例模式的本质

单例模式的本质是:控制实例数目

4.2 何时选用

当需要控制一个类的实例只能有一个,而且客户只能从一个全局访问点访问它时,可以选用单例模式,这些功能恰好是单例模式要解决的问题

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351

推荐阅读更多精彩内容