Параллельное программирование на С++ в действии. Практика разработки многопоточных программ | страница 19
Разумеется, перед тем задумываться о применении платформенно-зависимых средств, стоит как следует разобраться в том, что предлагает стандартная библиотека, поэтому начнем с примера.
1.4. В начале пути
Итак, вы получили новенький, с пылу с жару компилятор, совместимый со стандартом С++11. Что дальше? Как выглядит многопоточная программа на С++? Да примерно так же, как любая другая программа, — с переменными, классами и функциями. Единственное существенное отличие состоит в том, что некоторые функции могут работать параллельно, поэтому нужно следить за тем, чтобы доступ к разделяемым данным был безопасен (см. главу 3). Понятно, что для параллельного исполнения необходимо использовать специальные функции и объекты, предназначенные для управления потоками.
1.4.1. Здравствуй, параллельный мир
Начнем с классического примера — программы, которая печатает фразу «Здравствуй, мир». Ниже приведена тривиальная однопоточная программа такого сорта, от нее мы будем отталкиваться при переходе к нескольким потокам.
>#include
>int main() {
> std::cout << "Здравствуй, мир\n";
>}
Эта программа всего лишь выводит строку Здравствуй мир в стандартный поток вывода. Сравним ее с простой программой «Здравствуй, параллельный мир», показанной в листинге 1.1, — в ней для вывода сообщения запускается отдельный поток.
>#include
>#include
(1)
>void hello() ←
(2)
>{
> std::cout << "Здравствуй, параллельный мир\n";
>}
>int
>main() {
> std::thread t(hello); ←
(3)
> t.join(); ←
(4)
>}
Прежде всего, отметим наличие дополнительной директивы >#include
(1). Все объявления, необходимые для поддержки многопоточности, помещены в новые заголовочные файлы; функции и классы для управления потоками объявлены в файле >
, а те, что нужны для защиты разделяемых данных, — в других заголовках.
Далее, код вывода сообщения перемещен в отдельную функцию (2). Это объясняется тем, что в каждом потоке должна быть начальная функция, в которой начинается исполнение потока. Для первого потока в приложении таковой является >main()
, а для всех остальных задается в конструкторе объекта >std::thread
. В данном случае в качестве начальной функции объекта типа >std::thread
, названного >t
(3), выступает функция >hello()
.
Есть и еще одно отличие вместо того, чтобы сразу писать на стандартный вывод или вызывать