先日から愛機の調子が思わしくありません。突然フリーズ/リセットかかるんです。頻度はさほどに多くはないけどおっかなくて仕方がない、そこそこ動いているうちにもう1台新調することにしました。こんなこともあろうかと以前から少しずつ揃えていたパーツたちのおかげで予算にかなりの余裕があり、思い切ってイマドキのi7-2600Kに16GBのメモリを積んだゴキゲンな2号機ができあがりました。目下作業環境の引越中です。
CPUを奮発したおかげで論理8コア、8つのスレッドが同時に走る素敵な環境。遊ばせておくのはもったいなく、しばらく遊んでほったらかしになっていたインテルTBB(Threading Building Blocks)を再度試してみたくなりました。TBBはこれまでに何度かネタにしてきましたが、今回はスレッドの同期(Synchronization)のお話です。
前準備:スレッドの起こしかた
TBBでのスレッドの生成と起動はとっても簡単、関数オブジェクト:funを引数にtbb::tbb_threadのインスタンスを生成すれば、その時点でスレッドを生成し、そのスレッドの中でfun()が実行されます。引数のある関数オブジェクトの場合、引数が2つまでならtbb_threadのコンストラクト時に与えることができます。また、スレッドの終了を待つにはjoin()を呼び出します。
#include <tbb/tbb.h>
#include <iostream>
using namespace std;
// function
void sigma_function(int n, int* result) {
  int sum = 0;
  for ( int i = 1; i <= n; ++i ) {
    sum += i;
  }
  *result = sum;
}
// class
class sigma {
  int result_;
public:
  void operator()(int n) {
    int sum = 0;
    for ( int i = 1; i <= n; ++i ) {
      sum += i;
    }
    result_ = sum;
  }
  int result() const { return result_; }
};
int main() {
  // 関数をスレッドに
  int sum_f;
  tbb::tbb_thread thread_f(&sigma_function, 10, &sum_f);
  // クラスをスレッドに
  sigma sigma_object;
  tbb::tbb_thread thread_o( sigma_object,   10);
  // lambda
  auto sigma_lambda = [](int n, int* result) {
    int sum = 0;
    for ( int i = 1; i <= n; ++i ) {
      sum += i;
    }
    *result = sum;
  };
  // lambda式をスレッドに
  int sum_l;
  tbb::tbb_thread thread_l( sigma_lambda,   10, &sum_l);
  // 終了を待って出力
  thread_f.join();
  thread_o.join();
  thread_l.join();
  cout << sum_f << ' ' << sigma_object.result() << ' ' << sum_l << endl;
}
