for (int i=0; i < 100; ++i) grp.create_thread(worker);
grp.join_all();
std::cout << с.i_ << '\n'; // c.i = i
}
Совместно используемый ресурс должен где-то инициализироваться, и, возможно, вы захотите, чтобы это сделал тот поток, который первым стал его использовать. Переменная типа once_flag
(реальный ее тип зависит от платформы) и функция call_once
могут предотвратить повторную инициализацию объекта. Вам придется сделать две вещи.
Во-первых, проинициализируйте вашу переменную once_flag
с помощью макропеременной BOOST_ONCE_INIT
. Значение этой макропеременной зависит от платформы. В примере 12.5 класс Conn
представляет собой некоторое соединение (базы данных, сокета, оборудования и т.д.), которое я хочу инициализировать лишь однажды, несмотря на то, что несколько потоков могут пытаться сделать то же самое. Подобная ситуация возникает довольно часто, когда требуется динамически загружать библиотеку, имя которой может быть задано, например, в конфигурационном файле приложения. Флаг once_flag
— это переменная статического класса, потому что мне требуется только однократная инициализация независимо от количества существующих экземпляров этого класса. Поэтому я следующим образом устанавливаю этот флаг в начальное значение BOOST_ONCE_INIT
.
boost::once_flag Conn::initFlag_ = BOOST_ONCE_INIT;
Затем в моей рабочей функции я вызываю call_once
, которая синхронизирует доступ к моему инициализированному флагу и, следовательно, предотвращает параллельное выполнение другой инициализации. Я передаю в call_once
два аргумента:
boost::call_once(Conn::init, Conn::initFlag_);
Первым аргументом является адрес функции, которая будет выполнять инициализацию. Второй аргумент — это флаг. В данном случае несколько потоков могут попытаться выполнить инициализацию, но только первый в этом преуспеет.
12.5. Передача аргумента функции потока
Требуется передать аргумент в вашу функцию потока, однако средствами библиотеки Boost Threads предусматривается передача только функторов без аргументов.
Создайте адаптер функтора, который принимает ваши параметры и возвращает функтор без параметров. Адаптер функтора можно использовать там, где должен был бы быть функтор потока. Пример 12.6 показывает, как это можно сделать.
#include
#include
#include
#include
// typedef используется для того, чтобы приводимые ниже объявления лучше
// читались
typedef void (*WorkerFunPtr)(const std::string&);
template
typename ParamT> // тип ее параметра
struct Adapter {
Adapter(FunT f, ParamT& p) : // Сконструировать данный адаптер и
f_(f), p_(&p) {} // установить члены на значение функции и ее
// аргумента
void operator()() { // Просто вызов функции с ее аргументом
f_(*p_);
}
private:
FunT f_;
ParamT* p_; // Использовать адрес параметра. чтобы избежать лишнего
// копирования
};
void worker(const std::string& s) {
std::cout << s << '\n';
}
int main() {
std::string s1 = "This is the first thread!";
std::string s2 = "This is the second thread!";
boost::thread thr1(Adapter
boost::thread thr2(Adapter
thr1.join();
thr2.join();
}
Здесь приходится решать принципиальную проблему, причем характерную не только для потоков или проекта Boost, а общую проблему, возникающую при необходимости передачи функтора с одной сигнатурой туда, где требуется другая сигнатура. Решение состоит в создании адаптера.
Синтаксис может показаться немного путаным, но фактически в примере 12.6 создается временный функтор, который может вызываться конструктором потока как функция без аргументов (требуется именно такая функция). Но прежде всего используйте typedef
, чтобы указатель функции лучше воспринимался в тексте.
typedef void (*WorkerFunPtr)(const std::string&);