c++:自己动手实现线程安全的c++单例类

  前段时间使用c++做项目开发,需要根据根据配置文件路径加载全局配置文件,并对外提供唯一访问点。面对这样一个需求,自然的就想到了使用单例模式来创建一个单例配置对象,供外部调用。一开始想使用boost中自带的单例类来实现,但是遗憾的是,boost中的的单例类好像只能使用无参的类构造函数,而我希望将配置文件路径作为单例配置对象的构造函数参数,此外正好借此机会使用c++自己动手实现一个单例类。

  1.线程安全的c++单例类

   实现线程安全的c++单例类,主要要实现以下几点:1)构造函数私有化,即构造函数、拷贝构造函数和复制构造函数定义为private。构造函数私有化是为了防止在类外部定义类对象;拷贝构造函数私有化是为了防止拷贝行为产生多个实例;复制构造函数私有化,防止赋值产生多个实例。  2)提供静态全局访问点,供外部调用访问   3)通过锁机制或者static初始化,保证多线程访问单例对象安全。程序如下:

  清单1:单例类 config.h

 1 #ifndef _CONFIG_H_
 2 #define _CONFIG_H_
 3 #include <windows.h>
 4 #include <iostream>
 5 using namespace std;
 6 class Config
 7 {
 8 private:     //1.构造函数私有
 9     Config()
10     {
11         m_path = "config.cfg";
12         loadGlobalConfig();
13     }
14     Config(string path) :m_path(path)
15     {
16         loadGlobalConfig();
17     }
18     Config(const Config &);   //拷贝构造函数不实现,防止拷贝产生多个实例
19     Config & operator =  (const Config &);  //复制构造函数不实现,防止赋值产生多个实例
20 public:
21     static Config * getInstance()    //2.提供全局访问点
22     {
23         static Config m_singletonConfig;    //3.c++11保证了多线程安全,程序退出时,释放资源
24         return &m_singletonConfig;
25     }
26     void loadGlobalConfig()
27     {
28         //std::cout<<"111"<<std::endl;
29         //Sleep(1000);   //休眠1000ms
30         //std::cout<<"222"<<std::endl;
31         //加载配置文件......
32     }
33 private:
34     string m_path;    //配置文件的路径
35 };
36 #endif // _CONFIG_H_

  2. static线程安全测试

    前面提到,c++11保证了static对象在执行构造函数初始化时的线程安全性。对此c++11中的static变量的该特性,我做了一个实验,验证了static类对象的构造函数线程安全性。撤销清单1中28-30行代码的注释,执行main.cpp。main.cpp代码如下: 

  清单2 :main.cpp

 1 #include "config.h"
 2 #include <thread>
 3 #define THREAD_NUM 2
 4 void gTestStatic()
 5 {
 6     Config *pConf=Config::getInstance();
 7 }
 8 int main()
 9 {
10     std::thread threadArray[THREAD_NUM];
11     for (int i=0;i<THREAD_NUM;i++)
12     {
13         threadArray[i] = std::thread(&gTestStatic);
14     }
15     for (int i = 0; i < THREAD_NUM; i++)
16     {
17         threadArray[i].join();    //主线程等待所有的线程结束
18     }
19     return 0;
20 }

  清单3 : 实验结果

1 output:
2             111
3             222

  从这个实验可以看出,一个线程在执行类的构造函数时休眠1ms,另一个线程在等待,因此static对象的构造函数确实只执行了一次。因此,c++11确实保证了static对象构造函数初始化的多线程安全。

时间: 2024-10-11 23:24:04

c++:自己动手实现线程安全的c++单例类的相关文章

线程安全的懒汉式单例设计模式

首先回顾一下单利设计模式: 单利设计模式是通过某种方式使某个类只能创建一个对象实例供外界使用. 单利设计模式分为懒汉式和饿汉式: 饿汉式是线程安全的: 1 //饿汉式单利设计模式: 2 class Single{ 3 private static final Single single = new Single(); 4 private Single(){} 5 public static Single getInstance(){ 6 return s; 7 } 8 } 饿汉式实现线程安全,可

(五十七)线程的资源共享、单例的实现

[资源共享的问题] 例如线程A与B均实现数字Num的加一操作,如果不加以限制,可能A和B先后拿到最初的Num,然后返回Num+1,无法实现Num+1之后再+1. [互斥锁] 使用互斥锁(@synchronized)来解决,让线程A操作时锁住Num,不允许B的读写,直到A操作完并且写回后,再让B进行工作,再锁住Num,直到B操作完毕,再解开锁,类似于上厕所,在厕所内要锁门一样. 使用@synchronized(self){......}包装在内的为互斥锁的作用范围,会严重降低效率.因此应尽可能的减

iOS-主线程刷新UI【+单例】

主线程刷新UI dispatch_async(dispatch_get_main_queue(), ^{ /// }); 单例 static Tools *_sharedManger; @implementation Tools + (SingleTools *)sharedManger{ static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ if (!_sharedManger) { _sharedManger = [[T

【黑马】程序员————多线程(二)单例设计模式、线程间通信,JDK1.5互斥锁

------Java培训.Android培训.iOS培训..Net培训.期待与您交流!----- 一.单例设计模式 单例设计模式的意义: A.保证类在内存中只有一个对象,不提供外部访问方式,构造函数用private修饰. B.提供公共方法(static修饰,类的静态方法),获取类的实例.单例设计模式分为饿汉和懒汉两种模式. 饿汉式&懒汉式 class Test33 { public static void main(String[] args) { Fanjianan.getInstance()

Spring单例与线程安全小结

转:http://www.cnblogs.com/doit8791/p/4093808.html 一.Spring单例模式与线程安全 Spring框架里的bean,或者说组件,获取实例的时候都是默认的单例模式,这是在多线程开发的时候要尤其注意的地方. 单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 当多用户同时请求一个服务时,容器会给每一个请求分配一个线程,这是多个线程会并发执行该请求多对应的业务逻辑(成员方法),此时

java设计模式--解决单例设计模式中懒汉式线程安全问题

首先写个单例,懒汉模式: public class SingleDemo { private static SingleDemo s = null; private SingleDemo(){} public static SingleDemo getInstance(){ if(s == null){ s = new SingleDemo(); } return s; } } 写个测试类: public class ThreadDemo3 { public static void main(S

解决单例设计模式中懒汉式线程安全问题

首先写个单例: public class SingleDemo { private static SingleDemo s = null; private SingleDemo(){} public static SingleDemo getInstance(){ if(s == null){ s = new SingleDemo(); } return s; } } 写个测试类: public class ThreadDemo3 { public static void main(String

单例设计模式-(你确定自己写的懒汉单例真的是线程安全的吗)

1.单例设计模式的优缺点 优点: 1):只创建一个实例,就可以到处使用,加快创建实体的效率 缺点: 1):如果使用的频率比较低,实例会一直占据着内存空间,会造成资源浪费 2):可能会出现线程安全问题 2.单例设计模式的两种定法(饿汉.懒汉) 饿汉方法写法:(可能会造成资源浪费,类一被加载就创建了实例,但并不能确保这个实例什么时候会被用上) package com.zluo.pattern; /** * * 项目名称:single-instance <br> * 类名称:SingleInstan

C#单例测试(懒汉式双锁保证线程安全)

单例模式的概念 单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 关键点: 这个类只有一个实例,这是最基本的 它必须自行创建这个实例,外部不能实例化 进程内唯一 代码 using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace WCF_Host_Service { /// <summary>