c++ - decltype(..., void()) 和 void_t 的区别

标签 c++ sfinae

上次我找到了很多关于 SFINAE 的答案,这些答案建议使用 void_t 助手。

但我似乎不明白它有什么特别之处:

decltype (..., void()).

考虑这个例子:

template <typename...>
using void_t = void;

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

template <typename T>
struct has_foo <T, decltype (T().foo(), void())> : std::true_type {};

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

template <typename T>
struct has_bar <T, void_t <decltype (T().bar())> > : std::true_type {};

class MyClass1
{
public:
    int foo() { return 3; }
};

class MyClass2
{
public:
    double bar() { return 5.4; }
};

int main() {

    std::cout << has_foo<MyClass1>::value << std::endl;
    std::cout << has_foo<MyClass2>::value << std::endl;
    std::cout << has_bar<MyClass1>::value << std::endl;
    std::cout << has_bar<MyClass2>::value << std::endl;

    return 0;
}

两个特征的输出都符合预期,这让我认为这两个实现是相同的。我错过了什么吗?

最佳答案

这是表达同一事物的一种更具表现力、更简单的方式。

就是这样。

关于c++ - decltype(..., void()) 和 void_t 的区别,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35986886/

相关文章:

c++ - 是否可以检查是否为类定义了成员函数,即使成员是从未知基类继承的

c++ - 使用 SFINAE 原理时重载函数有歧义

c++ - 为什么模板实例化永远在这里进行?

c++ - 实现 std::variant 转换构造函数 - 或者:如何从参数包中找到从任何 T 到 Ti 的所有转换的第一个重载

c++ - 为什么我们应该在全局函数/对象上使用 "::"运算符?

c++ - 说明嵌入式软件中设计模式的使用

c++ - boost 不同的工具集

c++ - 使用 memoized 函数观察到奇怪的性能

c++ - Boost库中用于动态位集的硬件支持的popcount

c++ - 方法,调用所有基类的同名方法(如果存在)并将返回值保存到列表中