4

次のコードを検討してください。

#include <type_traits>
#include <iostream>

struct test1 {
    void Invoke() {};
};

struct test2 {
    template<typename> void Invoke() {};
};


enum class InvokableKind {
    NOT_INVOKABLE,
    INVOKABLE_FUNCTION,
    INVOKABLE_FUNCTION_TEMPLATE
};

template<typename Functor, class Enable = void>
struct get_invokable_kind {
    const static InvokableKind value = InvokableKind::NOT_INVOKABLE;
};

template<typename Functor>
struct get_invokable_kind<
  Functor,
  decltype(Functor().Invoke())
  >
{
    const static InvokableKind value = InvokableKind::INVOKABLE_FUNCTION;
};

template<typename Functor>
struct get_invokable_kind<
  Functor,
  decltype(Functor().Invoke<void>())
  >
{
    const static InvokableKind value = InvokableKind::INVOKABLE_FUNCTION_TEMPLATE;
};


int main() {
    using namespace std;

    cout << (get_invokable_kind<test1>::value == InvokableKind::INVOKABLE_FUNCTION) << endl;
    cout << (get_invokable_kind<test2>::value == InvokableKind::INVOKABLE_FUNCTION_TEMPLATE) << endl;

}

私がやろうとしているのは、「呼び出し可能性」の特定の定義をテストするためのメタ関数を作成することです。そして今、GCC 4.5.3でこのコンパイル エラーが発生しました。

prog.cpp:37:3: エラー: テンプレート引数 2 が無効です

どういう意味ですか?を専門にできるのに、専門にdecltype(Functor().Invoke())できないのはなぜdecltype(Functor().Invoke<void>())ですか?

4

1 に答える 1

6

template解析のあいまいさのため、次のように修飾する必要があります。

 decltype(Functor().template Invoke<void>())
                    ^^^^^^^^

関連: 「template」および「typename」キーワードをどこに、なぜ入れなければならないのですか?

また、コンストラクターstd::declvalではなく使用を検討してください。Functor()

于 2012-11-12T13:53:06.973 に答える