1. ホーム
  2. c++

[解決済み] C++11でのstd::threadへの参照によるオブジェクトの受け渡し

2023-07-31 20:54:11

質問

を作成する際に、オブジェクトを参照渡しできないのはなぜですか? std::thread ?

例えば、以下のようなスニピットはコンパイルエラーになります。

#include <iostream>
#include <thread>

using namespace std;

static void SimpleThread(int& a)  // compile error
//static void SimpleThread(int a)     // OK
{
    cout << __PRETTY_FUNCTION__ << ":" << a << endl;
}

int main()
{
    int a = 6;

    auto thread1 = std::thread(SimpleThread, a);

    thread1.join();
    return 0;
}

エラーです。

In file included from /usr/include/c++/4.8/thread:39:0,
                 from ./std_thread_refs.cpp:5:
/usr/include/c++/4.8/functional: In instantiation of ‘struct std::_Bind_simple<void (*(int))(int&)>’:
/usr/include/c++/4.8/thread:137:47:   required from ‘std::thread::thread(_Callable&&, _Args&& ...) [with _Callable = void (&)(int&); _Args = {int&}]’
./std_thread_refs.cpp:19:47:   required from here
/usr/include/c++/4.8/functional:1697:61: error: no type named ‘type’ in ‘class std::result_of<void (*(int))(int&)>’
       typedef typename result_of<_Callable(_Args...)>::type result_type;
                                                             ^
/usr/include/c++/4.8/functional:1727:9: error: no type named ‘type’ in ‘class std::result_of<void (*(int))(int&)>’
         _M_invoke(_Index_tuple<_Indices...>)
         ^

ポインタを渡すように変更したのですが、もっと良い回避策はないでしょうか?

どのように解決するのですか?

スレッドを明示的に初期化し reference_wrapper を使って std::ref :

auto thread1 = std::thread(SimpleThread, std::ref(a));

(または std::cref の代わりに std::ref に変更しました。) のメモによると に関する cppreference を参照してください。 std:thread :

スレッド関数への引数は、値によって移動またはコピーされます。参照引数をスレッド関数に渡す必要がある場合、それはラップされなければなりません (例えば std::ref または std::cref ).