2
#include <thread>
#include <cassert>
#include <iostream>
#include <string>
#include <future>
#include <utility>

void ThFun(std::promise<std::string>&& prms)
{
    std::string hello = "Hello From Future!\n";
    prms.set_value(hello);
}

int main()
{
    std::promise<std::string> prms;
    auto ftr = prms.get_future();
    std::thread th(&ThFun, std::move(prms));
    std::cout << "Hello from Main\n";
    std::string str = ftr.get();
    std::cout << str << std::endl;
    th.join();

    return 0;
}

このビルドエラーが発生します。

1>c:\program files (x86)\microsoft visual studio 11.0\vc\include\functional(1140):     error C2664: 'void (std::promise<_Ty> &&)' : cannot convert parameter 1 from 'std::promise<_Ty>' to 'std::promise<_Ty> &&'
1>          with
1>          [
1>              _Ty=std::string
1>          ]
1>          You cannot bind an lvalue to an rvalue reference
1>          c:\program files (x86)\microsoft visual studio 11.0\vc\include\functional(1140) : while compiling class template member function 'void std::_Bind<_Forced,_Ret,_Fun,_V0_t,_V1_t,_V2_t,_V3_t,_V4_t,_V5_t,<unnamed-symbol>>::operator ()(void)'
1>          with
1>          [
1>              _Forced=true,
1>              _Ret=void,
1>              _Fun=void (__cdecl *const )(std::promise<std::string> &&),
1>              _V0_t=std::promise<std::string>,
1>              _V1_t=std::_Nil,
1>              _V2_t=std::_Nil,
1>              _V3_t=std::_Nil,
1>              _V4_t=std::_Nil,
1>              _V5_t=std::_Nil,
1>              <unnamed-symbol>=std::_Nil
1>          ]
1>          c:\program files (x86)\microsoft visual studio 11.0\vc\include\thread(50) : see reference to class template instantiation 'std::_Bind<_Forced,_Ret,_Fun,_V0_t,_V1_t,_V2_t,_V3_t,_V4_t,_V5_t,<unnamed-symbol>>' being compiled
1>          with
1>          [
1>              _Forced=true,
1>              _Ret=void,
1>              _Fun=void (__cdecl *const )(std::promise<std::string> &&),
1>              _V0_t=std::promise<std::string>,
1>              _V1_t=std::_Nil,
1>              _V2_t=std::_Nil,
1>              _V3_t=std::_Nil,
1>              _V4_t=std::_Nil,
1>              _V5_t=std::_Nil,
1>              <unnamed-symbol>=std::_Nil
1>          ]
1>          c:\users\jim\documents\visual studio 11\projects\promisesandfutures    \promisesandfutures\main.cpp(18) : see reference to function template instantiation 'std::thread::thread<void(__cdecl *)(std::promise<_Ty> &&),std::promise<_Ty>>(_Fn,_V0_t &&)' being compiled
1>          with
1>          [
1>              _Ty=std::string,
1>              _Fn=void (__cdecl *)(std::promise<std::string> &&),
1>              _V0_t=std::promise<std::string>
1>          ]

私はVC11を使用しています。エラーは左辺値をバインドできないことを示していますが、私はそれを右辺値にするためにstd::moveを使用しています。

ありがとう。

std::refの編集は正常に機能します。

4

1 に答える 1

7

エラーメッセージから、VC11はgcc4.7より前のgccと同じ間違いを犯したようstd::bindですstd::threadstd::bind引数がコピー可能である必要がありますが、そうでstd::threadはありません。これは、などの移動専用タイプstd::promiseを引数としてスレッドに渡すことができないことを意味します。

実際、あなたの例はgcc 4.7、またはMSVC2010とC++11スレッドライブラリの私のjust:: thread実装で動作しますが、ネイティブライブラリを使用するとgcc 4.6で同様のエラーメッセージが表示されて失敗します(gcc4.6は次の場合にも動作します) just :: threadを使用)。

于 2012-04-19T12:15:31.333 に答える