如何将std :: enable_if与自推理返回类型一起使用?

时间:2013-12-20 18:33:00

标签: c++ template-meta-programming typetraits c++14 enable-if

C++14将具有可根据返回值推断其返回类型的函数。

auto function(){
    return "hello world";
}

我可以通过返回类型习惯用法将这种行为应用于使用enable_if作为SFINAE的函数吗?

例如,让我们考虑以下两个功能:

#include <type_traits>
#include <iostream>

//This function is chosen when an integral type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_integral<T>::value>::type {
    std::cout << "integral" << std::endl;
    return;
}

//This function is chosen when a floating point type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_floating_point<T>::value>::type{
    std::cout << "floating" << std::endl;
    return;
}

int main(){

  function(1);    //prints "integral"
  function(3.14); //prints "floating"

}

如您所见,使用SFINAE按返回类型惯用法选择正确的函数。 但是,这些都是无效功能。 enable_if的第二个参数默认设置为void。这将是相同的:

//This function is chosen when an integral type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_integral<T>::value, void>::type {
    std::cout << "integral" << std::endl;
    return;
}

//This function is chosen when a floating point type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_floating_point<T>::value, void>::type{
    std::cout << "floating" << std::endl;
    return;
}

我可以对这两个函数做些什么,以便它们的返回类型由返回值推断出来吗?

gcc 4.8.2(使用--std=c++1y

4 个答案:

答案 0 :(得分:11)

std::enable_if不必在返回类型中,从C ++ 11开始,它可以是模板参数的一部分。

所以你的等效函数可以是(或者,这样的效果):

enum class enabler_t {};

template<typename T>
using EnableIf = typename std::enable_if<T::value, enabler_t>::type;

//This function is chosen when an integral type is passed in
template<class T, EnableIf<std::is_integral<T>>...>
auto function(T t) {
    std::cout << "integral" << std::endl;
    return;
}

//This function is chosen when a floating point type is passed in
template<class T, EnableIf<std::is_floating_point<T>>...>
auto function(T t) {
    std::cout << "floating" << std::endl;
    return;
}

它也可以是函数中的参数:

//This function is chosen when an integral type is passed in
template<class T>
auto function(T t, EnableIf<std::is_integral<T>>* = nullptr) {
    std::cout << "integral" << std::endl;
    return;
}

//This function is chosen when a floating point type is passed in
template<class T>
auto function(T t, EnableIf<std::is_floating_point<T>>* = nullptr) {
    std::cout << "floating" << std::endl;
    return;
}

这将保留自动类型扣除和SFINAE。

答案 1 :(得分:6)

std::enable_if可以是返回类型,函数参数或模板参数。如果使用返回类型或模板参数,则会出现函数重定义错误,因此需要使用std::enable_if作为函数参数:

#include <type_traits>
#include <iostream>

template<class T, typename = typename std::enable_if<std::is_integral<T>::value, void>::type>
auto function(T t, typename std::enable_if<std::is_integral<T>::value, void>::type* dummy = nullptr) {
    std::cout << "integral" << std::endl;
    return 0;
}

//This function is chosen when a floating point type is passed in
template<class T, typename = typename std::enable_if<std::is_floating_point<T>::value, void>::type>
auto function(T t, typename std::enable_if<std::is_floating_point<T>::value, void>::type* dummy = nullptr) {
    std::cout << "floating" << std::endl;
    return 0.0f;
}

int main() 
{
    auto ret = function(0); // integral
    auto ret2 = function(0.0f); // floating
    std::cout << std::boolalpha << std::is_integral<decltype(ret)>::value << std::endl; // true
    std::cout << std::is_floating_point<decltype(ret2)>::value << std::endl; // true
}

答案 2 :(得分:3)

在@ user1508519的答案行中,我们可以从方法中删除enable_if参数,并仅将其保留为模板参数。 我们依赖enable_if<false>未定义type这一事实,因此enable_if<false>::type(不存在)是SFINAE的一个很好的工具 - 在方法签名中,包括模板参数。

无需在方法本身中实际使用此模板参数!

因此:

template<class T,
typename std::enable_if<std::is_integral<T>::value>::type* dummy = nullptr>
auto function(T t) {
    std::cout << "integral" << std::endl;
    return 0;
}

// This function is chosen when a floating point type is passed in
template<class T, 
typename std::enable_if<std::is_floating_point<T>::value>::type* dummy = nullptr>
auto function(T t) {
    std::cout << "floating" << std::endl;
    return 0.0f;
}

int main() {
    auto ret = function(0); // integral
    auto ret2 = function(0.0f); // floating
    cout << std::boolalpha;
    cout << std::is_integral<decltype(ret)>::value << endl; // true
    cout << std::is_floating_point<decltype(ret2)>::value << endl; // true
}

输出

integral
floating
true
true

答案 3 :(得分:0)

正如其他地方所提到的,std::enable_if可以用来形成一个返回类型;一个函数参数;或模板参数。

然而,后两种方法的优点在于它们改变了相关功能或对象的签名。另一方面,在返回类型中使用std::enable_if会使函数和模板参数计数保持不变。

使用C ++ 11中的lambda返回类型的自动推断(可能扩展到C ++ 14中的普通函数),如果有一种技术可以推断返回类型,那将是理想的。 在返回类型上使用std::enable_if。让你的蛋糕吃 - 几乎。唉,目前看来这是不可能的。