前两天,一个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_h class 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++ 单例一些需要注意的地方:

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

C++ 安全单例模式总结的更多相关文章

  1. C++实现线程安全的单例模式

    在某些应用环境下面,一个类只允许有一个实例,这就是著名的单例模式.单例模式分为懒汉模式,跟饿汉模式两种. 首先给出饿汉模式的实现 template <class T> class sing ...

  2. 23种设计模式--单例模式-Singleton

    一.单例模式的介绍 单例模式简单说就是掌握系统的至高点,在程序中只实例化一次,这样就是单例模式,在系统比如说你是该系统的登录的第多少人,还有数据库的连接池等地方会使用,单例模式是最简单,最常用的模式之 ...

  3. angular2系列教程(十)两种启动方法、两个路由服务、引用类型和单例模式的妙用

    今天我们要讲的是ng2的路由系统. 例子

  4. java设计模式之--单例模式

    前言:最近看完<java多线程编程核心技术>一书后,对第六章的单例模式和多线程这章颇有兴趣,我知道我看完书还是记不住多少的,写篇博客记录自己所学的只是还是很有必要的,学习贵在坚持. 单例模 ...

  5. 设计模式C#合集--单例模式

    单例模式 代码: 第一种: private static Singleton singleton = null; private Singleton() { } public static Singl ...

  6. 设计模式之单例模式(Singleton)

    设计模式之单例模式(Singleton) 设计模式是前辈的一些经验总结之后的精髓,学习设计模式可以针对不同的问题给出更加优雅的解答 单例模式可分为俩种:懒汉模式和饿汉模式.俩种模式分别有不同的优势和缺 ...

  7. GOF23设计模式之单例模式

    ·核心作用: -保证一个类只有一个实例,并且提供一个访问该实例的全局访问点. ·常见应用场景: -Windows的Task Manager(任务管理器)就是很典型的单例模式 -Windows的Recy ...

  8. GJM : C#设计模式(1)——单例模式

    感谢您的阅读.喜欢的.有用的就请大哥大嫂们高抬贵手"推荐一下"吧!你的精神支持是博主强大的写作动力以及转载收藏动力.欢迎转载! 版权声明:本文原创发表于 [请点击连接前往] ,未经 ...

  9. PHP设计模式(四)单例模式(Singleton For PHP)

    今天讲单例设计模式,这种设计模式和工厂模式一样,用的非常非常多,同时单例模式比较容易的一种设计模式. 一.什么是单例设计模式 单例模式,也叫单子模式,是一种常用的软件设计模式.在应用这个模式时,单例对 ...

  10. java设计模式之单例模式(几种写法及比较)

    概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...

随机推荐

  1. Linux目录与文件的权限

    零.Linux中的权限为什么重要? 权限直接关系数据安全! 一.用户基础概念: 所有者(owner):拥有这个文件的用户.一般拥有目录或文件的所有权限. 用户组(group):几个用户组成一个用户组, ...

  2. [ABP实战开源项目]--YoYoCms目录

    ABP是"ASP.NET Boilerplate Project (ASP.NET样板项目)"的简称. ASP.NET Boilerplate是一个用最佳实践和流行技术开发现代WE ...

  3. mac下安装MySQL完整步骤(图文详情)

    原文摘自:http://www.jb51.net/article/103841.htm 最近使用Mac系统,准备搭建一套本地web服务器环境.因为Mac系统自带PHP和apach,但是没有自带mysq ...

  4. 在Caffe上运行Cifar10示例

    准备数据集 在终端上运行以下指令: cd caffe/data/cifar10 ./get_cifar10.sh cd caffe/examples/cifar10 ./create_cifar10. ...

  5. Node.js~ioredis处理耗时请求时连接数瀑增

    回到目录 关于redis连接数过高的解释 对于node.js开发环境里,使用传统的redis或者使用ioredis都是不错的选择,而在处理大数据请求程中,偶尔出现了连接池( redis服务端的最大可用 ...

  6. JS 中new一个对象发生了什么事

    今天看到一个360的前端面试题: function A(){}function B(a){  this.a=a;}function C(a){  if(a){    this.a=a;   }}A.p ...

  7. kotlin, 一种新的android平台一级开发语言

    最近看到一则科技新闻, 大致内容是google将kotlin语言作为android应用开发的一级语言, 与java并驾齐驱, 这是一个开发界的大事件大新闻, 连google的亲儿子go语言也没有这种待 ...

  8. 1、在eclipse中导入Java的jar包方法---JDBC【图文说明】

    1.Eclipse环境下jar包导入 在Eclipse环境下编写Java程序,常常会借用到各种jar包.如:连接数据库时,导入jar包是必须的.导入方法如下: 1.打开eclipse,右击要导入jar ...

  9. 谷歌发布基于机器学习的Android APP安全检测系统:Google Play Protect

    Google Play作为众所周知的在线应用市场,因为审查制度的松散,经常会有一些恶意软件伪装成其他应用混入其中.此前阿里聚安全小编就报道了2例关于恶意软件伪装在Google Play上的事件:< ...

  10. 我来说说XML文件中的xmlns、xmlns:xsi和xsi:schemaLocation的具体含义

      文章摘自:https://yq.aliyun.com/articles/40353               http://www.cnblogs.com/zhao1949/p/5652167. ...