浅谈Singleton

我们在实现单例模式时,考虑的第一点是对象的创建和访问只有一个共同的入口,所以必须满足以下两个条件:

  • 访问入口是与对象无关的
  • 构造函数必须是隐藏的,不能被外部通过直接调用产生新对象

所以我们的实现中有两个关键点:

  • 静态方法getInstance做为访问入口
  • 将构造函数设为private

根据这两个原则,我们先有了第一版简单实现:

V1.0

//Singleton.h
class Singleton{
public:
    static Singleton* getInstance();
private:
    Singleton(){};
    static Singleton _instance;
};

//Singleton.cpp
Singleton Singleton::_instace;
Singleton* Singleton::getInstance(){
    return &_instance;
}

这里貌似是把Singleton实现的很好,完全符合上述提到的两点。但是“一切代码都没有完美的”,我们总能找出缺点和问题的。比如这里,_instance对象是声明为static的,即在程序运行之初就初始化在静态存储区域了。但是万一我的整个程序中根本就没用到这个对象呢?这时候这就是一种浪费了。就算是在程序中用到了,我也希望是在我第一次使用它时,它才被创建并初始化的。在《Effective C++》的条款04中,Scott Meyers更是指出了这种实现方式的严重的错误之处:
假设有两个Singleton类,其中一个以另一个为参数进行初始化,即存在一个 static SingletonA SingletonA::_instance 和另一个以此为参数初始化的 static SingletonB SingletonB::_instance(SingletonA::_instance) ,这两个non-local static对象位于两个不同的编译单元内,而C++对于“定义于不同的编译单元内的non-local static对象”的初始化相对次序并无明确定义。所以就会出现SingletonB::_instance对象在SingletonA::_instance对象之前初始化,那就意味着此时SingletonA::_instance处于“半随机”状态,这会导致不可测知的程序行为。
所以我们有了下一个版本:

V1.1

//Singleton.h
class Singleton{
public:
    static Singleton* getInstance();
private:
    Singleton(){};
};

//Singleton.cpp
Singleton* Singleton::getInstance(){
    static Singleton _instance;
    return &_instance;
}

这也是Scott Meyers在《Effective C++》中提出的方法——用local static替换non-local static, 这样不仅解决了初始化次序问题,同时实现了lazy initialization。当然,“一切代码都没有完美的”,考虑到多线程系统中与初始化相关的race conditions:当多个线程同时第一次调用getInstance(),此时会多次产生静态对象。Scott Meyers提出的解决方法是,在程序的单线程启动阶段,手工的调用getInstance()方法。这并不是一个优雅的写法,我们希望只在真正需要对象的时候才去调用getInstance()。那么就需要在getInstance方法里加锁保护。
所以我们又有了下一个版本:

V1.2

//Singleton.h
#include <mutex>
class Singleton{
public:
    static Singleton* getInstance();
private:
    Singleton(){};
    static std::mutex _mutex;
};

//Singleton.cpp
std::mutex Singleton::_mutex;
Singleton* Singleton::getInstance(){
    std::lock_guard<std::mutex> lck(_mutex);
    static Singleton _instance;
    return &_instance;
}

这里又引入了一个新的non-local static变量Singleton::_mutex,是不是感觉是在拆东墙补西墙?幸好的是这里无需面对初始化次序的问题,Singleton::_mutex的初始化不依赖其他静态对象,而lazy initialization在这里也不是什么大问题,也就顾不上了。
虽然通过加锁互斥解决了静态变量初始化的线程安全问题,但是考虑另一种调用情形:当_instance已经在之前的调用中构造出来了,然后此时再次调用getInstance()方法时,仍然需要对锁进行操作,这样显然是不合理的。应该在判断_instance已经存在并初始化之后就直接返回其指针,不做锁操作。那么就需要再引入一个non-local static变量来表示是否已初始化过,OMG!那么干脆就用一个指针来指代_instance对象吧,这样也就可以用指针是否为NULL来判断对象是否已创建。
下面是我们的判断对象是否已创建的指针版本:

V2.0

//Singleton.h
#include <mutex>
class Singleton{
public:
    static Singleton* getInstance();
private:
    Singleton(){};
    static std::mutex _mutex;
    static Singleton* _instance;
};

//Singleton.cpp
std::mutex Singleton::_mutex;
Singleton* Singleton::_instance = NULL;
Singleton* Singleton::getInstance(){
    if (NULL == _instance){
        std::lock_guard<std::mutex> lck(_mutex);
        _instance = new Singleton();
    }
    return _instance;
}

这样好像是把问题都解决了。慢着,当两个线程同时第一次调用getInstance时,都会判断 NULL == _instance 条件成立,于是都要执行构造对象操作,由于锁的存在,同时只会有一个线程构造对象,但是当前一个线程构造完毕之后就退了锁,于是第二个线程进去了,继续重新构造对象。看来只有在加锁之后对 NULL == _instance 条件再进行一次判断了。
加锁前后两次判断的版本:

V2.1

//Singleton.h
#include <mutex>
class Singleton{
public:
    static Singleton* getInstance();
private:
    Singleton(){};
    static std::mutex _mutex;
    static Singleton* _instance;
};

//Singleton.cpp
std::mutex Singleton::_mutex;
Singleton* Singleton::_instance = NULL;
Singleton* Singleton::getInstance(){
    if (NULL == _instance){
        std::lock_guard<std::mutex> lck(_mutex);
        if (NULL == _instance){
            _instance = new Singleton();
        }
    }
    return _instance;
}

这种方式称之为 Double-Checked Locking 技术。那么,这个方式还有没有改进的空间呢?当然,“一切代码都没有完美的”,试想我们的系统中有很多类是Singleton模式时,难道我们要把这套方法对每个类都复制一遍么?为了避免代码重复,我们怎么去实现呢?继承?抑或者类模板?

路漫漫其修远兮,吾将上下而求索

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

推荐阅读更多精彩内容

  • 前言 本文主要参考 那些年,我们一起写过的“单例模式”。 何为单例模式? 顾名思义,单例模式就是保证一个类仅有一个...
    tandeneck阅读 2,529评论 1 8
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,807评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,729评论 18 399
  • 1 场景问题# 1.1 读取配置文件的内容## 考虑这样一个应用,读取配置文件的内容。 很多应用项目,都有与应用相...
    七寸知架构阅读 6,860评论 12 68
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,330评论 11 349