前两天,一个C++ 的单例实现又掉坑里了。做好一个安全的单例模式可并不简单。这里总结一下C++ 的几个单例实现方案。

1. 函数静态变量法

利用单例函数的静态变量,实现单例构造。代码如下:

class StaticVarSingleTon {
public:static StaticVarSingleTon *GetInstance() {static StaticVarSingleTon s_instance;return &s_instance;}private:StaticVarSingleTon() {}virtual ~StaticVarSingleTon() {};StaticVarSingleTon(const StaticVarSingleTon &);StaticVarSingleTon& operator=(const StaticVarSingleTon& other);
};

这里利用函数的静态变量,只会存在一份的特性,来实现单例的构造。代码直接明了。

优点

  • 代码简单,直接明了
  • 还是个懒加载模式

缺点

  • 静态变量的构造,不是线程安全的。

2. 类静态成员变量

利用类的静态变量的全局唯一性,来实现单例的构造。代码如下:

//
//  StaticMemberSingleton.h
//#ifndef StaticMemberSingleton_h
#define StaticMemberSingleton_hclass StaticMemberSingleTon {
public:static StaticMemberSingleTon *GetInstance() {return &s_instance;}private:StaticMemberSingleTon() {}virtual ~StaticMemberSingleTon() {};StaticMemberSingleTon(const StaticMemberSingleTon &);StaticMemberSingleTon& operator=(const StaticMemberSingleTon& other);private:static StaticMemberSingleTon s_instance;
};#endif /* StaticMemberSingleton_h*/

//
//  StaticMemberSingleton.cpp
//#include "StaticMemberSingleton.h"StaticMemberSingleTon StaticMemberSingleTon::s_instance;

优点

  • 这里的StaticMemberSingleTon StaticMemberSingleTon::s_instance 是一个全局变量。只会出现一份。
  • 全局变量的初始化,在main 函数执行之前完成。可以保证线程安全。

缺点

  • 当有另外一个 StaticMemberSingletonB,在构造函数中依赖 StaticMemberSingletonA 的单例对象时,可能出现StaticMemberSingletonA 的单例对象还没有初始化的问题。

让我们用代码来验证一下,我们构造两个单例:StaticMemberSingletonA, StaticMemberSingletonB.

StaticMemberSingletonA 的构造函数,调用StaticMemberSingletonB 的方法;

StaticMemberSingletonB 的构造函数,调用StaticMemberSingletonA 的方法。

代码如下。

//
//  StaticMemberSingletonA.h
//#ifndef StaticMemberSingletonA_h
#define StaticMemberSingletonA_h#include <stdio.h>class StaticMemberSingleTonA {
public:static StaticMemberSingleTonA *GetInstance() {return &s_instance;}void showValue() {printf("SingleTonA value %d\n", value);}private:StaticMemberSingleTonA();virtual ~StaticMemberSingleTonA() {};StaticMemberSingleTonA(const StaticMemberSingleTonA &);StaticMemberSingleTonA& operator=(const StaticMemberSingleTonA& other);private:static StaticMemberSingleTonA s_instance;private:int value = 0;
};#endif /* StaticMemberSingletonA_h*/

//
//  StaticMemberSingletonA.cpp
//#include "StaticMemberSingletonA.h"
#include "StaticMemberSingletonB.h"StaticMemberSingleTonA StaticMemberSingleTonA::s_instance;StaticMemberSingleTonA::StaticMemberSingleTonA() {value = 1;StaticMemberSingleTonB::GetInstance()->showValue();
}

//
//  StaticMemberSingletonB.h
//#ifndef StaticMemberSingletonB_h
#define StaticMemberSingletonB_h#include <stdio.h>class StaticMemberSingleTonB {
public:static StaticMemberSingleTonB *GetInstance() {return &s_instance;}void showValue() {printf("SingleTonB value %d\n", value);}private:StaticMemberSingleTonB();virtual ~StaticMemberSingleTonB() {};StaticMemberSingleTonB(const StaticMemberSingleTonB &);StaticMemberSingleTonB& operator=(const StaticMemberSingleTonB& other);private:static StaticMemberSingleTonB s_instance;private:int value = 0;
};#endif /* StaticMemberSingletonB_h*/

//
//  StaticMemberSingletonB.cpp
//#include "StaticMemberSingletonB.h"
#include "StaticMemberSingletonA.h"StaticMemberSingleTonB StaticMemberSingleTonB::s_instance;StaticMemberSingleTonB::StaticMemberSingleTonB() {value = 2;StaticMemberSingleTonA::GetInstance()->showValue();
}

//
//  main.cpp
//#include <stdio.h>
#include "StaticMemberSingletonA.h"int main(int argc, const char * argv[]) {StaticMemberSingleTonA::GetInstance();return 0;
}

执行一下,结果如下:

SingleTonA value 0
SingleTonB value 2
Program ended with exit code: 0

3. 线程安全的单例方法

一般常见的C++ 线程安全的单例实现代码,如下:

//
//  SafeSingleton.h
//#ifndef SafeSingleton_h
#define SafeSingleton_h#include "Mutex.h"class SafeSingleton {
public:static SafeSingleton *GetInstance();private:SafeSingleton() {};virtual ~SafeSingleton() {};SafeSingleton(const SafeSingleton &);SafeSingleton& operator=(const SafeSingleton& other);private:static SafeSingleton *s_instance;static Mutex          s_insMutex;
};#endif /* SafeSingleton_h*/

//
//  SafeSingleton.cpp
//#include "SafeSingleton.h"SafeSingleton *SafeSingleton::s_instance;
Mutex          SafeSingleton::s_insMutex;SafeSingleton *SafeSingleton::GetInstance() {if (s_instance == nullptr) {s_insMutex.lock();if (s_instance == nullptr) {s_instance = new SafeSingleton();}s_insMutex.unlock();}return s_instance;
}

注意:

  • 第一次判断 s_instance 非空,是为了提升性能,避免无谓的加锁。
  • 获得锁后,必须再次判断 s_instance 非空,避免多线程下二次创建。
  • 另外,由于所有实例的构造,都在main函数之后执行了。而锁对象是全局变量,在main 之前就已经完成初始化了,不会出现方案2 中的对象未初始化现象。
  • 当然,如果真这儿做了,会出现死锁。

4. 还未结束

我们在C++ 层实现了一个网络状态监控模块,这个模块给iOS 业务层使用。当时业务层实现了自己的一个网络状态模块。大致代码如下所示:

@implementation IOSNetworkState+ (void)load {[IOSNetworkState sharedInstance];
}+ (instancetype)sharedInstance {static IOSNetworkState *instance;static dispatch_once_t onceToken;dispatch_once(&onceToken, ^{instance = [[IOSNetworkState alloc] init];});return instance;
}- (instancetype)init {self = [super init];if (self) {Network::NetworkMonitor::GetInstance()->DoXXX();}return self;
}

然后就悲剧了,APP 起来就crash。crash 的位置是,执行 Network::NetworkMonitor::GetInstance() 方法时,加锁操作crash。原因是Mutex 对象未初始化。

原来,OC 类的 +(void)load 方法,其执行时期是类的加载期。比全局对象(就是我们的Mutex)的初始化要早。当然这个时候,main 函数更加没有得到执行。

自然我们这时候,执行加锁操作就会引发异常了。

5. 总结

简单总结一下,使用c++ 单例一些需要注意的地方:

  • 一:使用线程安全的单例方法。
  • 二:尽量避免在单例类的构造方法中,使用其他的单例对象。
  • 三:不要在类的加载期方法中,使用其他单例对象。其实,在类加载期方法中,不应该涉及业务处理。

转载于:https://www.cnblogs.com/rivermoon/p/7072564.html

C++ 安全单例模式总结相关推荐

  1. Java单例模式个人总结(实例变量和类变量)

    Java单例模式 背景知识:Static关键字. 在对于定义类的变量,分为两种,是否具有static修饰的变量: 没有static修饰的变量,通过类的实例化(对象)引用,改变量称为实例变量: 使用st ...

  2. GOF23设计模式(创建型模式)单例模式

    目录: 一:单例模式的核心作用.常见应用场景 二:五种单例模式及其实现 三:关于反射和反序列化破解单例模式的漏洞,以及相应的解决方案 四:测试五种单例模式的效率 一:核心作用及常见应用场景: 核心作用 ...

  3. Java设计模式:单例模式

    学而时习,稳固而之心, 好久没有复习java的知识了,今天有空温习了单例模式,这里记录一下 单例模式是常见的设计模式的一种,其特点就是 指一个类只有一个实例,且该类能自行创建这个实例  , 保证一个类 ...

  4. [Python设计模式] 第21章 计划生育——单例模式

    github地址:https://github.com/cheesezh/python_design_patterns 单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式 ...

  5. Python 精选笔试面试习题—sorted 与 sort 单例模式、统计字符个数Count、垃圾回收、lambda函数、静态方法、类方法、实例方法、分布式锁、

    1. 字典根据键从小到大排序? In[38]: dic = {"name": "Tom", "age": 30, "country ...

  6. 单例模式的两种实现方式对比:DCL (double check idiom)双重检查 和 lazy initialization holder class(静态内部类)...

    首先这两种方式都是延迟初始化机制,就是当要用到的时候再去初始化. 但是Effective Java书中说过:除非绝对必要,否则就不要这么做. 1. DCL (double checked lockin ...

  7. 基础设计模式:单例模式+工厂模式+注册树模式

    单例模式: 通过提供自身共享实例的访问,单例设计模式用于限制特定对象只能被创建一次. 使用场景: 一般数据库实例都会用单例模式 实现: 单例设计模式就是要一个类只能实例化一个对象. 要想让一个类只能实 ...

  8. 设计模式——单例模式(Singleton)

    保证一个类仅有一个实例,并提供一个访问它的全局访问点.--DP UML类图 模式说明 个人认为单例模式是所有设计模式中最为简单的一个模式,因为实现这个模式仅需一个类,而不像其他模式需要若干个类.这个模 ...

  9. 设计模式 之美 -- 单例模式

    为什么要使用单例? 一个类只允许创建一个对象或者实例. 背景简介:使用多线程并发访问同一个类,为了保证类的线程安全,可以有两种方法: 将该类定义为单例模式,即该类仅允许创建一个实例 为该类的成员函数添 ...

  10. 【C++】C/C++ 中的单例模式

    目录 part 0:单例模式3种经典的实现方式 Meyer's Singleton Meyers Singleton版本二 Lazy Singleton Eager Singleton Testing ...

最新文章

  1. 【硬核书】矩阵代数基础
  2. Vector用法(C++ Primer中文版)
  3. 人的执念真的是非常的可怕
  4. Nginx的启动、停止
  5. 大话数据结构12 串String
  6. Linux下安装Octave
  7. SCRIPT鼠标放在图片上才显示色彩 平时是灰色的图片
  8. [Android系列—] 1. Android 开发环境搭建与Hello World
  9. C Tricks(十四)—— 余数
  10. 第 7 章 Neutron - 069 - 理解 Neutron Server 分层模型
  11. Matlab 谢尔宾斯基三角形
  12. 哈哈哈哈,IT总监VS美发总监
  13. Linux系统下利用共享内存模拟迅雷下载
  14. 翻遍互联网都找不到的解决方案,一行代码轻松实现 Gitbook 默认折叠左侧菜单效果
  15. 倒排索引、正排索引,以及ElasticSearch对倒排索引的优化方法
  16. 移植tslib,测试电容屏
  17. 什么是video codec? video codec在实际业务的应用。
  18. 【DL with Pytorch】第 3 章 :使用 DNN 的分类问题
  19. walking机器人入门教程-深度学习-使用yolov5进行物体识别
  20. android热成像模块,Arduino制作简易热成像装置

热门文章

  1. opencv+Dlib人脸融合
  2. ubuntu16.04命令行模式和图形界面互相切换
  3. 末转变者登录服务器一直在排队,魔兽世界:国服神级服务器,排队持续两年,哈霍兰有何特别之处?...
  4. java反射 获取方法_java反射——获取类的方法信息
  5. fastreport按条件查询_数据分析sql从入门到精通--条件和过滤查询
  6. kubernetes视频教程笔记 (5)-网络通讯方式
  7. php session失效
  8. SpringBoot+thymeleaf 发送邮件
  9. 设计模式之GOF23代理模式02
  10. centos8 挂载ntfs_Centos 挂载Ntfs分区