0

promise を使用して、packaged_task をテンプレート クラスとして実装しようとしています。

私のコンパイル エラーは、削除された関数を参照していると言っています。コピーおよび/または移動のセマンティクスを実装する必要があると思われますが、どのように、どこから始めればよいか混乱しています。どんなアドバイスでも大歓迎です:

#include "stdafx.h"
#include <iostream>
#include <future>
#include <functional>
#include <thread>
using namespace std;

//Base case
template<class>
class promised_task;

//Templated class
template<class Ret, class...Args>
class promised_task<Ret(Args...)> {
public:
    //Constructor
    //Takes a function argument that is forwarded to fn member
    template<class F>
    explicit promised_task(F&& f) :fn(f){}

    //get_future member function:
    future<Ret> get_future(){
        return prom.get_future();
    }

    //Set value
    void operator()(Args&&...args){
        prom.set_value(fn(forward<Args>(args)...));
    }

private:
    //Promise member
    promise<Ret> prom;

    //Function member
    function<Ret(Args...)> fn;
};


//Sample function from cplusplus.com
int countdown(int from, int to){
    for (int i = from; i != to; --i){
        cout << i << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }
    cout << "Lift off!" << endl;
    return from - to;
}

//Verification function also from cplusplus.com
int main(){
    promised_task<int(int, int)>tsk(countdown);
    future<int>ret = tsk.get_future();

    thread th(move(tsk), 10, 0);

    int value = ret.get();

    cout << "The countdown lasted for " << value << " seconds." << endl;

    th.join();

    cout << "Press any key to continue:" << endl;
    cin.ignore();
    return 0;
}
4

1 に答える 1

1
thread th(move(tsk), 10, 0)

これがエラーを生成する行だと思います。

追加:

promised_task(promised_task&& o):
  prom(std::move(o).prom), fn(std::move(o).fn)
{}

promised_task手動で move ctor を記述するため。C++11 では、コンパイラが上記の移動コンストラクタ (または同等のもの) を作成する必要があります。MSVC2013 は準拠した C++11 コンパイラではありません。

エラーは、コピー コンストラクターがないため暗黙的に削除されたpromised_taskvia を移動できないことを訴えています。コンパイラは.promised_task(promised_task const&)promisepromised_task(promised_task&&)

型を移動しようとしたときに、移動操作がない場合は、copy が暗黙的に呼び出されます。コピーが削除/不可能/アクセス不能の場合、タイプをコピーできないというエラーが表示されます。

MSVC2015 はその欠陥を修正していることに注意してください。C++11 の MSVC2015 実装に残っている大きな穴は、Microsoft が「式 SFINAE」と呼んでいるものと、ノックオン効果 (実装に必要なためにコンプライアンスに失敗するライブラリ コンポーネント) です。

C++11 への準拠に関する最近の連絡で、MSVC2015 サイクルのいずれかの時点で最終消費者向けの更新プログラム (ベータ版ではありません) で出荷する予定であると述べていますが、ブロックされたライブラリ機能は次のメジャー リリースまで有効にしないでください ( ODR 違反を避けるため)。

于 2015-05-04T18:10:01.700 に答える