关于c++11中static类对象构造函数线程安全的验证

  在c++11中,static静态类对象在执行构造函数进行初始化的过程是线程安全的,有了这个特征,我们可以自己动手轻松的实现单例类,关于如何实现线程安全的单例类,请查看c++:自己动手实现线程安全的c++单例类。下面给出一个验证例子:

1.清单1:staticSafe.h

 1 #include <iostream>
 2 #include <thread>
 3 using namespace std;
 4 class Cnum
 5 {
 6 public:
 7     Cnum()
 8     {
 9         std::cout << "construct start" << std::endl;
10         std::this_thread::sleep_for(std::chrono::seconds(5));   //构造函数中休眠5s
11         num++;
12         std::cout << "construct stop" << std::endl;
13     }
14     void Test()
15     {
16         std::cout << "id:" << std::this_thread::get_id()<<",num="<< num<<std::endl;    
17         std::this_thread::sleep_for(std::chrono::seconds(1));
18     }
19     static int num;
20 };

2. 清单2:main.cpp

 1 int Cnum::num = 0;     //初始化静态成员
 2 void func(void* argv)
 3 {
 4     int i = *(int *)argv;
 5     std::cout <<i<< std::endl;   //每个线程启动时,打印线程次序号
 6     static Cnum a;   //内部静态成员变量的初始化时线程安全的,只有一个线程能执行初始化,其他线程会在此阻塞
 7     a.Test();
 8 }
 9 void main()
10 {     
11     for (int i = 0; i < 10; i++)    //开启10个线程,模拟并发访问情形
12     {
13             std::thread t1(func, (void *)&i);                     
14             t1.detach(); 
15     }
16     system("pause ");//暂停一下
17 }

3. 运行结果

  

  从结果中可以看出:1)当打印“construct stop” 以后,其他线程才陆续执行Test函数  2) 所有线程打印的num为1。以上现象表明:当有一个线程执行static类对象的构造函数时,其他欲访问该static类对象的线程都是阻塞的。

 

  

 

posted @ 2017-05-22 09:36  南宫轩诺  阅读(4549)  评论(0编辑  收藏  举报