从一个经典的例子开始:一个打印“Hello World.”的程序。一个非常简单的在单线程中运行的Hello World程序如下所示,当我们谈到多线程时,它可以作为一个基准。
#include<iostream> using namespace std; int main() { cout<<"Hello World"<<endl; return 0; }
这个程序所做的就是将“Hello World”写进标准输出流。让我们将它与下面清单所示的简单的“Hello, Concurrent World”程序做个比较,它启动了一个独立的线程来显示这个信息。
一个简单的Hello, Concurrent World程序:
#include<iostream> #include<thread> ① void hello() ② { cout<<"Hello World"<<endl; } using namespace std; int main() { thread t(hrllo); ③ t.join(); ④//这句代码是为了让main线程等待t线程结束再结束,如果不加则有可能会使得main线程比t线程先运行结束,那么t线程就没法运行也就没有输出了 }
第一个区别是增加了 #include <thread> ①,标准C++库中对多线程支持的声明在新的头文件中:管理线程的函数和类在 <thread> 中声明,而保护共享数据的函数和类在其他头文件中声明。
其次,写信息的代码被移动到了一个独立的函数中②。因为每个线程都必须具有一个初始函数(initial function),新线程的执行在这里开始。对于应用程序来说,初始线程是main(),但是对于其他线程,可以在 std::thread 对象的构造函数中指定——在本例中,被命名为t③的 std::thread 对象拥有新函数hello()作为其初始函数。
下一个区别:与直接写入标准输出或是从main()调用hello()不同,该程序启动了一个全新的线程来实现,将线程数量一分为二——初始线程始于main(),而新线程始于hello()。
新的线程启动之后③,初始线程继续执行。如果它不等待新线程结束,它就将自顾自地继续运行到main()的结束,从而结束程序——有可能发生在新线程运行之前。这就是为什么在④这里调用 join() 的原因——详见第2章,这会导致调用线程(在main()中)等待与 std::thread 对象相关联的线程,即这个例子中的t。
如果这看起来仅仅为了将一条信息写入标准输出而做了大量的工作,一般来说并不值得为了如此简单的任务而使用多线程,尤其是在这期间初始线程并没做什么.