关于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类对象的线程都是阻塞的。