C++设计模式_12_Singleton 单件模式

2023-10-26 09:45

本文主要是介绍C++设计模式_12_Singleton 单件模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在之前的博文C++57个入门知识点_44:单例的实现与理解中,已经详细介绍了单例模式,并且根据其中内容,单例模式已经可以在日常编码中被使用,本文将会再做梳理。
Singleton 单件模式可以说是最简单的设计模式,但由于多线程环境的双检查锁里的内存reorder的问题,实现时的细节并不简单,大家需要注意多线程环境下的安全做法。为了整体的一致性,本篇都称之为单件模式。

文章目录

  • 1. “对象性能”模式
    • 1.1 典型模式
  • 2. 动机(Motivation)
  • 3. Singleton 单件模式的实现版本
    • 3.1 版本1:线程非安全版本
    • 3.2 版本2:线程安全版本,但锁的代价过高
    • 3.3 版本3:双检查锁
    • 3.4 版本4:C++ 11版本之后的跨平台实现
    • 3.5 总结
  • 4. 模式定义
  • 5. 结构
  • 6. 要点总结
  • 7. 其他参考

单件模式属于一个新的类别,将其归结到“对象性能”模式,该模式的简介如下:

1. “对象性能”模式

面向对象很好地解决了“抽象”的问题,但是不可避免地要付出一定的代价(虚函数(内存等代价哒)、继承)。对于通常情况来讲,面向对象的成本大都可以忽略不计。但是某些情况(倍乘效应等),面向对象所带来的成本必须谨慎处理。

1.1 典型模式

  • Singleton
  • Flyweight

2. 动机(Motivation)

  • 在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性、以及良好的效率
  • 如何绕过常规的构造器,提供一种机制来保证一个类只有一个实例?
  • 这应该是类设计者的责任,而不是使用者的责任

3. Singleton 单件模式的实现版本

首先将构造函数和拷贝构造函数设置为私有,如果不写的话,编译器会默认生成两个公有的,让外界不能使用它们,唯一的做法就是将其设置为私有的。然后再设置静态变量,并进行静态变量的初始化。代码如下

class Singleton{
private:Singleton();Singleton(const Singleton& other);
public:static Singleton* getInstance();static Singleton* m_instance;
};

3.1 版本1:线程非安全版本

先来看第一种写法,第一次调用的时候,m_instance == nullptr成立的情况下m_instance = new Singleton();创建对象,返回m_instance,第二次调用的话,m_instance == nullptr不成立,继续返回原先的对象,这样就可以保证永远只返回第一次的对象。

//线程非安全版本
Singleton* Singleton::getInstance() {if (m_instance == nullptr) {m_instance = new Singleton();}return m_instance;
}

上述的版本会有其他问题,它在单线程下是OK的,但是多线程环境下就不行了。假设我有2个thread,threadA进到if (m_instance == nullptr)判断第一次调用,还没执行下一行,threadB假如此时得到时间片,开始执行if (m_instance == nullptr)也是成立,就会执行下一行,此时再跳回threadA,此时就会继续执行下一行,这样就会产生2个对象,因此这是线程非安全的。为了实现多线程安全如何去做呢?

3.2 版本2:线程安全版本,但锁的代价过高

第一种常见做法:加锁

//线程安全版本,但锁的代价过高
Singleton* Singleton::getInstance() {Lock lock; //锁的局部变量,出作用域之后消失,释放锁if (m_instance == nullptr) {m_instance = new Singleton();}return m_instance;
}

TreadA进到Lock lock;开始加锁,往下执行时,另外一个线程threadB进来,但是TreadA已经获取到锁,仍未释放,threadB就需要一直等到TreadA执行完函数释放锁。等threadB进去之后m_instance == nullptr就不成立了。

3.3 版本3:双检查锁

上一种版本实现了锁的安全,但是锁的代价太高。假如对象已经创建,那么有两个线程同时访问,threadA执行到Lock lock;,threadB是无法进入的,threadB是一个读操作,当存在多个线程的情况下,读操作是不需要加锁的。读操作是没有安全问题的,这个时候加锁对于读操作的线程是一种等待时间上的浪费。假如高并发场景下,web服务是典型高并发,10万人同时访问时,锁对于读操作代价也会是十分高的。

所以发展出一种新的实现形式,double check lock 双检查锁,之前的版本是不管什么情况,先锁,现在是if(m_instance==nullptr)先问下变量是否为空值,如果为空才加锁,如果不空就不需要加锁,直接返回即可,加完锁之后再判断是否为空。这样就在读操作时减少了时间代价。

//双检查锁,但由于内存读写reorder不安全
Singleton* Singleton::getInstance() {if(m_instance==nullptr){Lock lock;if (m_instance == nullptr) {m_instance = new Singleton();}}return m_instance;
}

有人会考虑,第二个判断if (m_instance == nullptr) 是不是可以去掉,我们假设2个线程,threadA经过第一个判断进来,然后在执行Lock lock;之前,threadB也会进来到Lock lock;之前,两个线程即使加锁也就会等一等,都会执行代码m_instance = new Singleton();,这样就会有2个对象。第二个判断就是为了放置这种情况的出现。

上面的版本看起来就是没有问题的了,也相当一段时间内迷惑了计算机领域的专家,大概2000年时才被JAVA领域的专家发现漏洞:内存读写reorder导致双检查锁的失效。

什么是reorder,正常情况下,大家看到的代码会有指令序列,那么大家通常会认为,指令序列会按照你所想象的方式执行,但是实际上整个代码到了CPU的指令层次(线程是在指令层次抢时间片),指令和我们很多时候的假设很可能不一样。比如: m_instance = new Singleton();一般会有以下的假设,拆分成3个步骤,第一步是先分配内存,第二步是调用构造器(对分配的内存进行初始化),第三步把内存的地址(指针)给m_instance。但是这三步是我们假想的,但是在指令级别,很可能被编译器优化reorder,在reorder之后可能就会先第一步,第三步,最后再第三步。很多研究者在不同的CPU上做实验发现很多语言都会这种现象,这样一来就有些乱套。按照可能的reorder顺序,先分配内存,直接就赋值,m_instance就不是nullptr了,构造器还没调用,另一个线程做第一次判断时发现m_instance不是nullptr,直接返回一个对象,返回的这个对象还没有经过构造器处理,其状态不对,是不可用的。

3.4 版本4:C++ 11版本之后的跨平台实现

为了解决上面的问题,java和c#语言就加了volatile的关键字来保证按照我们的想法来执行。VC++中自己也加了volatile的关键字。

C++11之后才有了这样的机制,可以帮助可以在语言层面实现跨平台的实现。

//C++ 11版本之后的跨平台实现 (volatile)
std::atomic<Singleton*> Singleton::m_instance; //声明原子对象
std::mutex Singleton::m_mutex;Singleton* Singleton::getInstance() {Singleton* tmp = m_instance.load(std::memory_order_relaxed);//得到tmp指针,帮助实现屏蔽编译器的reorderstd::atomic_thread_fence(std::memory_order_acquire);//获取内存fence,内存reorder的屏障//tmp就不会被reorder,底下为双检查锁的操作if (tmp == nullptr) {std::lock_guard<std::mutex> lock(m_mutex);tmp = m_instance.load(std::memory_order_relaxed);if (tmp == nullptr) {tmp = new Singleton;std::atomic_thread_fence(std::memory_order_release);//释放内存fencem_instance.store(tmp, std::memory_order_relaxed);}}return tmp;
}

经过上面的操作,才在C++11之后实现跨平台的实现。

3.5 总结

总体来说,如果是单线程使用版本1已经是足够好了;如果是多线程,使用版本2是OK的,但是效率不高;双检查锁在不使用volatile不能用,是很容易出现问题的;C++11版本之后跨平台实现双检查锁就使用的是最后一个版本

4. 模式定义

保证一个类仅有一个实例,并提供一个该实例的全局访问点
–《设计模式》GoF

5. 结构

在这里插入图片描述

6. 要点总结

  • Singleton模式中的实例构造器可以设置为protected以允许子类派生
  • Singleton模式一般不要支持拷贝构造函数和Clone接口,因为这有可能导致多个对象实例,与Singleton模式的初衷违背。
  • 如何实现多线程环境下安全的Singleton?注意对双检查锁的正确实现。

7. 其他参考

C++设计模式——单例模式

这篇关于C++设计模式_12_Singleton 单件模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/285543

相关文章

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的