1

似乎替换失败有时是一个错误。
有人可以告诉我什么时候会出错,什么时候不会?在这里
查看运行结果 谢谢!

感谢您提供信息丰富的答案!
此代码无法使用 g++ 4.8 编译,但可以按预期使用 clang++ 3.2、g++ 4.7.3 和 intel 13.0.1。所以现在我确定这是 g++ 4.8 的一个错误。我已将此报告给 gcc bugzila。

In substitution of ‘template<class C> static constexpr int has<T>::test(decltype (sizeof (C:: x))) [with C = C; T = foo] [with C = foo]’:
required from ‘const int has<foo>::value’
required from here
error: invalid use of non-static member function ‘std::string foo::x()’

编码

template <typename T>
struct has {
    template <typename>
    constexpr static int test(...) {
      return 0;
    }
    template <typename C>
    constexpr static int test(decltype(sizeof(C::x))) {  // Doesn't compile.
      return 1;   // Is a member variable.
    }
    template <typename C, int c =
        sizeof(decltype(((C*)nullptr)->x()))>
    constexpr static int test(int) {
      return 2;   // Is a member function.
    }
    static const int value = test<T>(0);
};

struct foo {
    string x();
};
struct bar {
    string x;
};
int main() {
    std::cout << has<int>::value << std::endl;
    std::cout << has<foo>::value << std::endl;
    std::cout << has<bar>::value << std::endl;
}
4

1 回答 1

4

规则本质上是错误必须依赖于它直接所属的声明的模板参数。

这个例子看起来像一个编译器错误。它不能正确地排除特定的语法缺陷。

如果您希望它在类具有名为 的非静态成员函数时返回成功x,则应使用&C::x,因为您可以获取指向成员函数的指针,但不能简单地将其命名为独立的子表达式。

我首选的方法是

template< typename, typename = void >
struct has_x
    : std::false_type {};

template< typename t >
struct has_x< t, typename std::enable_if< & t::x == & t::x >::type >
    : std::true_type {};
于 2013-04-15T13:57:38.467 回答