C++学习之路(四):线程安全的单例模式
(一)简单介绍
单例模式分为两种类型:懒汉模式和饿汉模式。
懒汉模式:在实际类对象被调用时才会产生一个新的类实例,并在之后返回这个实例。多线程环境下,多线程可能会同时调用接口函数创建新的实例,为了防止这种并发情况,需要采用锁机制来保证线程安全性,但由于多线程抢占互斥锁,可能会导致系统的性能瓶颈。
饿汉模式:在程序开始时就会产生一个该类的实例,并在之后返回这个实例。由于在进入主函数之前就由主线程完成了对象的初始化化,所以不用担心多线程问题,也不存在线程安全问题。
(二)懒汉模式举例
#include <iostream>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h> using namespace std; class Singleton
{
private:
Singleton()
{
cout<<"class create"<<endl;
}
static Singleton *m_instance;
static pthread_mutex_t mutex;
class Garbo
{
public:
~Garbo()
{
if (Singleton::m_instance)
{
cout<<"class end"<<endl;
delete Singleton::m_instance;
}
}
};
static Garbo garbo;
public:
static Singleton *getInstance();
// ~Singleton()
// {
// cout<<"class end"<<endl;
// delete m_instance;
// }
}; Singleton* Singleton::m_instance = nullptr;
pthread_mutex_t Singleton::mutex = PTHREAD_MUTEX_INITIALIZER;
Singleton::Garbo Singleton::garbo; Singleton* Singleton::getInstance()
{
if (m_instance == nullptr)
{
pthread_mutex_lock(&mutex);
if (m_instance == nullptr)
m_instance = new Singleton();
pthread_mutex_unlock(&mutex);
}
return m_instance;
} int main(void)
{
Singleton *singleton = Singleton::getInstance();
return ;
}
上述代码中,在类实例初始化时才调用getInstance创建一个新实例,为了保证线程安全(即互斥),定义互斥锁。将类构造函数定义为private,防止类外部直接调用构造函数。
两次加锁:
假设现在有多个线程欲创建新对象,当前m_instance为nulltpr,进入第一层if,所有线程抢占互斥锁,其中一个线程完成对类实例的创建后解锁并返回,此时唤醒其他线程,当前m_instance已被创建,其他线程将不予以创建新实例,直接返回。从这里也可以看出,由于线程之间对锁的抢占,会有一定的性能开销。
关于单例模式的析构函数:
由于singleton对象是一个static对象,直接delete singleton是不能够调用类析构函数的(注释内容),要想释放m_instance实例,需要内嵌一个class,并初始该静态类成员,当类成员释放时调用其析构函数从而释放m_instance实例。
另一种写法:
测试代码:
class Singleton
{
private:
Singleton(){}
public:
static Singleton* getInstance()
{
Lock(); //not needed after C++0X
static Singleton instacne;
Unlock; //not needed after C++0X
return instance;
}
}
上述在C++0X以后,要求编译器保证内部静态变量的线程安全性,可以不加锁。但在C++0X以前,仍然需要加锁。
(三)饿汉模式举例
在程序开始时就会产生一个类实例,并在以后返回该实例。
测试代码:
class Singleton
{
private:
Singleton(){}
static Singleton *m_instance;
public:
static Singleton *getInstance(void)
{
return m_instance;
}
}; Singleton* Singleton::m_instance = new Singleton();
C++学习之路(四):线程安全的单例模式的更多相关文章
- Redis——学习之路四(初识主从配置)
首先我们配置一台master服务器,两台slave服务器.master服务器配置就是默认配置 端口为6379,添加就一个密码CeshiPassword,然后启动master服务器. 两台slave服务 ...
- zigbee学习之路(四):按键控制(中断方式)
一.前言 通过上次的学习,我们学习了如何用按键控制led,但是在实际应用中,这种查询方式占用了cpu的时间,如果通过中断控制就可以解决这个问题,我们今天就来学习按键控制的中断方式. 二.原理分析 传统 ...
- [原创]java WEB学习笔记79:Hibernate学习之路--- 四种对象的状态,session核心方法:save()方法,persist()方法,get() 和 load() 方法,update()方法,saveOrUpdate() 方法,merge() 方法,delete() 方法,evict(),hibernate 调用存储过程,hibernate 与 触发器协同工作
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- Qt 学习之路:线程总结
前面我们已经详细介绍过有关线程的一些值得注意的事项.现在我们开始对线程做一些总结. 有关线程,你可以做的是: 在QThread子类添加信号.这是绝对安全的,并且也是正确的(前面我们已经详细介绍过,发送 ...
- Qt 学习之路:线程和 QObject
前面两个章节我们从事件循环和线程类库两个角度阐述有关线程的问题.本章我们将深入线程间得交互,探讨线程和QObject之间的关系.在某种程度上,这才是多线程编程真正需要注意的问题. 现在我们已经讨论过事 ...
- Qt 学习之路:线程和事件循环
前面一章我们简单介绍了如何使用QThread实现线程.现在我们开始详细介绍如何“正确”编写多线程程序.我们这里的大部分内容来自于Qt的一篇Wiki文档,有兴趣的童鞋可以去看原文. 在介绍在以前,我们要 ...
- Qt 学习之路 :线程简介
现代的程序中,使用线程的概率应该大于进程.特别是在多核时代,随着 CPU 主频的提升,受制于发热量的限制,CPU 散热问题已经进入瓶颈,另辟蹊径地提高程序运行效率就是使用线程,充分利用多核的优势.有关 ...
- python学习之路 四 :文件处理
本节重点 掌握文件的读.写.修改方法 掌握文件的处理模式的区别 一.文件读取 1.读取全部内容 # 一次性读取文件 f = open("test.txt",'r',en ...
- Java学习之路(四):面向对象
Java中的面向对象 概念:面向对象的原本的意思是“”万物皆对象“” 面向对象思想的特点: 是一种更符合我们思想习惯的思想,将复杂的事情简单化 使我们角色发生了转换,将我们从执行者变成了指挥者 面向对 ...
- react.js学习之路四
针对学习react.js中,我感觉最大的疑惑点就是bind(this)的绑定和指向问题了,我被这个问题弄的头昏,有时候调用组件的时候,直接显示undefined,不存在这个组件,当时的心情是崩溃的,整 ...
随机推荐
- 第14天:逻辑运算符、if、for语句
今天学习了逻辑运算符.if.for语句基础知识. 一.逻辑运算符 1.&&(与) 一假即假,同真为真2.||(或)一真即真,同假为假3.!(非)切记:参与逻辑运算的,都是布尔值.也就是 ...
- error C2143: 语法错误 : 缺少“;”(在“类型”的前面)
C编程老是遇到这个问题: 错误 error C2143: 语法错误 : 缺少“;”(在“类型”的前面) d:\kinectproject\ceshiglad\ceshiglad\shili.c ces ...
- BZOJ 1042 硬币购物(背包DP+容斥原理)
可以看出这是个多重背包,运用单调队列优化可以使每次询问达到O(s).这样总复杂度为O(s*tot). 会TLE. 因为改题的特殊性,每个硬币的币值是不变的,变的只是每次询问的硬币个数. 我们不妨不考虑 ...
- Python 开篇及第一个Python程序
本节内容 python 简单介绍 python 2.x 或者python 3.x python 安装 第一个python程序 一.python简单介绍 python的创始人为吉多.范罗苏姆(Guido ...
- BZOJ4866 Ynoi2017由乃的商场之旅(莫队)
显然能重排为回文串相当于出现次数为奇数的字母不超过一个.考虑莫队,问题在于如何统计添加/删除一位的贡献.将各字母出现次数奇偶性看做二进制数,做一个前缀和一个后缀和.在右端添加一位时,更新区间的前缀.后 ...
- WPF 进度条ProgressBar
今天研究了一下wpf的进度条ProgressBar 1.传统ProgressBar WPF进度条ProgressBar 这个控件,如果直接写到循环里,会死掉,界面会卡死,不会有进度.需要把进度条放到单 ...
- (八)Redis事务与连接操作
事务: multi # 标记一个事务块的开始.事务块内的多条命令会按照先后顺序被放进一个队列当中,最后由EXEC命令原子性地执行 exec # 执行所有事务块内的命令 descard # 取消事务,放 ...
- 算法语言Scheme修订6报告 R6RS简体中文翻译
算法语言Scheme修订6报告 R6RS简体中文翻译 来源 https://r6rs.mrliu.org/ MICHAEL SPERBERR. KENT DYBVIG, MATTHEW FLATT ...
- [Leetcode] search in rotated sorted array ii 搜索旋转有序数组
Follow up for "Search in Rotated Sorted Array":What if duplicates are allowed? Would this ...
- 在CentOS 6.5 中安装JDK 1.7 + Eclipse并配置opencv的java开发环境(二)
一.安装JDK 1.7 1. 卸载OpenJDK rpm -qa | grep java rpm -e --nodeps java-1.6.0-openjdk-1.6.0.0-1.50.1.11.5. ...