先看一个最简单的教科书式单例模式:
class CSingleton
{
public:
static CSingleton* getInstance()
{
if (NULL == ps)
{//tag1
ps = new CSingleton;
}
return ps;
}
private:
CSingleton(){}
CSingleton & operator=(const CSingleton &s);
static CSingleton* ps;
};
CSingleton* CSingleton::ps = NULL;
有2个要点:
1.private的构造函数和=操作符,用于防止类外的实例化和被复制;
2.static的类指针和get方法。
在大多数单线程情况下,以上代码大都会运行得很好,除非遇到中断:
1.当程序运行到tag1 处触发了中断;
2.中断处理程序恰调用的也是getInstance函数。
可想而知,这和多线程的情况类似,假设线程A 运行到tag1处,还没来得及new,此时ps仍然是NULL,线程B(或中断处理程序) 同时也运行到此通过if判断,那么将会实例化2个CSingleton对象,显然是不对的。
为了解决上述问题,自然而然,最容易想到也最常用的方法是加锁,因此getInstance改成这样:
static CSingleton* getInstance()
{
lock();//伪代码
if (NULL == ps)
{
ps = new CSingleton;
}
return ps;
}
加了锁以后貌似解决了上述问题,但也同样带来了新的问题:如果程序到处是诸如:
CSingleton::instance()->aaaa(); CSingleton::instance()->bbbb(); CSingleton::instance()->cccc();
这样的调用,除了第一次的lock()有用外,后面的都是在做无用功,lock()的代价说大不大,但在某些情况下还是会提高程序延迟,这对追求完美的程序猿来说是完全无法接受的。
于是乎,咱想出了一个办法:
static CSingleton* getInstance()
{
if (NULL == ps)//这里加了次判断,只有第一次才会为true而调用lock()
{
lock();//伪代码
if (NULL == ps)
{
ps = new CSingleton;
}
}
return ps;
}
很久以后我才知道,这个方法有个很高大上的名字,叫做双重检查锁定模式,简称DCLP(Double Checked Locking Pattern)。
DCLP很好地解决了多次调用不必要的lock()。
然而,你们以为这样就完了?too young。。
DCLP在多线程下仍然存在2个根本问题:
1.程序的指令执行顺序不确定;
2.编译器优化问题。
先说2,在某些编译器下,以上的两个if判断只会执行一个,甚至一个都不执行,原因是编译器认为至少有一个if判断是多余的,它自动帮助我们优化了代码。
再说1,ps = new CSingleton; 这条语句会被拆分为这样的三个步骤执行:
1.为要new的对象开辟一块内存;
2.构造该对象,填入这块内存;
3.将ps指针指向这块内存。
以上三个步骤,2和3的顺序是不确定的,可能先2后3,也可能先3后2。。。
实际执行时可能是这样的:
static CSingleton* getInstance()
{
if (NULL == ps)
{
lock();//伪代码
if (NULL == ps)
{ //伪代码
ps = xx;//step 3
new sizeof(CSingleton);//step 1
new CSingleton;//step 2
}
}
return ps;
}
如果编译器按上述顺序执行代码,考虑如下状况:
线程A 执行到step 1还未执行后面的step 2,此时ps非空,但其指向的内存里面的内容还未被构造出来,于此同时线程B 进入这个函数,判断ps非空直接返回ps,但是调用者此时访问的ps内存实际内容CSingleton还没被构造呢,这是一块地址正确大小正确但内部数据不明的东西,当然会出错(调用者一般这么调用:CSingleton::getInstance()->aa(); CSingleton::getInstance()->bb(); CSingleton::getInstance()->cc();........此时的aa,bb,cc是啥玩意儿?)。
这也是为什么加上volatile关键字仍然不可以解决同步问题,volatile只解决了编译器优化问题,却无法控制机器指令执行顺序。
很遗憾的是,C/C++本身在设计时是不考虑多线程问题的,也就是说,要处理多线程问题还要程序猿自己想办法填坑。。
说了这么多,我们要讨论的问题仍然没有解决,庆幸的是,C++ 11提供了内存栅栏技术来解决这个问题,这里不赘述,有兴趣的读者可以自己搜索资料看看,不过是一些api调罢了。
那么,C++ 11 以前的代码如何解决这个问题呢?很不幸,并没有很好的解决方案,一种可行的方案是,程序中不要到处这么调用这个单例对象:
CSingleton::getInstance()->aa(); CSingleton::getInstance()->bb(); CSingleton::getInstance()->cc();
而是在程序开始就初始化缓存这个单例对象:
CSingleton* const g_ps = CSingleton::getInstance();//程序一开始就缓存这个单例对象 g_ps->aa(); g_ps->bb(); g_ps->cc();
但是如此带来的问题是程序一开始就实例化了这个单例对象,对象在整个程序的声明周期存在,这貌似叫饿汉式,而之前那种叫懒汉式,孰轻孰重,只有根据实际情况取舍了。
以上就是小编为大家带来的从C++单例模式到线程安全详解全部内容了,希望大家多多支持~
# c
# 单例模式
# 线程安全
# 详解C++实现线程安全的单例模式
# C++线程安全的单例模式讲解
# 老生常谈C++的单例模式与线程安全单例模式(懒汉/饿汉)
# 详解如何使用C++写一个线程安全的单例模式
# 很好
# 多线程
# 的是
# 是在
# 还没
# 解决了
# 这块
# 不确定
# 解决这个问题
# 这是
# 情况下
# 都不
# 有个
# 将会
# 是这样
# 调用者
# 说了
# 这么多
# 不可以
# 但在
相关文章:
企业宣传片制作网站有哪些,传媒公司怎么找企业宣传片项目?
网站制作话术技巧,网站推广做的好怎么话术?
简单实现Android验证码
如何通过服务器快速搭建网站?完整步骤解析
已有域名如何免费搭建网站?
建站之星如何优化SEO以实现高效排名?
家庭建站与云服务器建站,如何选择更优?
大学网站设计制作软件有哪些,如何将网站制作成自己app?
c# Task.ConfigureAwait(true) 在什么场景下是必须的
如何通过山东自助建站平台快速注册域名?
如何选择高性价比服务器搭建个人网站?
IOS倒计时设置UIButton标题title的抖动问题
网站制作与设计教程,如何制作一个企业网站,建设网站的基本步骤有哪些?
如何通过远程VPS快速搭建个人网站?
建站中国官网:模板定制+SEO优化+建站流程一站式指南
如何快速搭建高效WAP手机网站吸引移动用户?
电视网站制作tvbox接口,云海电视怎样自定义添加电视源?
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
深圳网站制作案例,网页的相关名词有哪些?
哈尔滨网站建设策划,哈尔滨电工证查询网站?
如何用y主机助手快速搭建网站?
如何在VPS电脑上快速搭建网站?
东莞专业制作网站的公司,东莞大学生网的网址是什么?
东莞市网站制作公司有哪些,东莞找工作用什么网站好?
如何在阿里云高效完成企业建站全流程?
如何撰写建站申请书?关键要点有哪些?
如何在云主机上快速搭建网站?
常州自助建站费用包含哪些项目?
建站之星安装后如何自定义网站颜色与字体?
c# 服务器GC和工作站GC的区别和设置
如何高效利用亚马逊云主机搭建企业网站?
建站之星官网登录失败?如何快速解决?
网站制作公司排行榜,抖音怎样做个人官方网站
Avalonia如何实现跨窗口通信 Avalonia窗口间数据传递
英语简历制作免费网站推荐,如何将简历翻译成英文?
宝塔面板如何快速创建新站点?
桂林网站制作公司有哪些,桂林马拉松怎么报名?
定制建站流程解析:需求评估与SEO优化功能开发指南
c# await 一个已经完成的Task会发生什么
如何正确下载安装西数主机建站助手?
建站之星与建站宝盒如何选择最佳方案?
制作网站的过程怎么写,用凡科建站如何制作自己的网站?
如何有效防御Web建站篡改攻击?
油猴 教程,油猴搜脚本为什么会网页无法显示?
微课制作网站有哪些,微课网怎么进?
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
Bpmn 2.0的XML文件怎么画流程图
c++怎么编写动态链接库dll_c++ __declspec(dllexport)导出与调用【方法】
*请认真填写需求信息,我们会在24小时内与您取得联系。