本节书摘来自异步社区出版社《C++并发编程实战》一书中的第1章,第1.4节,作者:【美】 Anthony Williams (威廉姆斯),更多章节内容可以访问云栖社区“异步社区”公众号查看。
1.4 开始入门
好,现在你有一个很棒的与C++11兼容的编译器。接下来呢?一个多线程C++程序是什么样子的?它看上去和其他所有C++程序一样,通常是变量、类以及函数的组合。唯一真正的区别在于某些函数可以并发运行,所以你需要确保共享数据的并发访问是安全的,详见第3章。当然,为了并发地运行函数,必须使用特定的函数以及对象来管理各个线程。
你好,并发世界
让我们从一个经典的例子开始:一个打印“Hello World.”的程序。一个非常简单的在单线程中运行的Hello, World程序如下所示,当我们谈到多线程时,它可以作为一个基准。
清单1.1这个程序所做的一切就是将“Hello World”写进标准输出流。让我们将它与下面清单所示的简单的Hello, Concurrent World程序做个比较,它启动了一个独立的线程来显示这个信息。
清单1.1 一个简单的Hello,Concurrent World程序
第一个区别是增加了#include<thread>
。在标准C++库中对多线程支持的声明在新的头文件中,用于管理线程的函数和类在<thread>
中声明,而那些保护共享数据的函数和类在其他头文件中声明。
其次,写信息的代码被移动到了一个独立的函数中。这是因为每个线程都必须具有一个初始函数(initialfunction
),新线程的执行在这里开始。对于应用程序来说,初始线程是main()
,但是对于所有其他线程,这在std::thread
对象的构造函数中指定——在本例中,被命名为t的std::thread对象拥有新函数hello()
作为其初始函数。
下一个区别,与直接写入标准输出或是从main()
调用hello()
不同,该程序启动了一个全新的线程来实现,将线程数量一分为二——初始线程始于main()
而新线程始于hello()
。
在新的线程启动之后,初始线程继续执行。如果它不等待新线程结束,它就将自顾自地继续运行到main()
的结束,从而结束程序——有可能发生在新线程有机会运行之前。这里调用join()
的原因——详见第2章,这会导致调用线程(在main()
中)等待与std::thread对象相关联的线程,即这个例子中的t。
如果这看起来像是仅仅为了将一条信息写入标准输出而做了大量的工作,那么它确实如此——正如上文1.2.3节所描述的,一般来说并不值得为了如此简单的任务而使用多线程,尤其是如果在这期间初始线程无所事事。在本书后面的内容中,我们将通过实例来展示在哪些情景下使用多线程可以获得明确的收益。