Выберите конструктор класса с помощью enable if

рассмотрим следующий код:

#include <iostream>
#include <type_traits>

template <typename T>
struct A {
    int val = 0;

    template <class = typename std::enable_if<T::value>::type>
    A(int n) : val(n) {};
    A(...) { }

    /* ... */
};

struct YES { constexpr static bool value = true; };
struct NO { constexpr static bool value = false; };

int main() {
    A<YES> y(10);
    A<NO> n;
    std::cout << "YES: " << y.val << std::endl
              << "NO:  " << n.val << std::endl;
}

Я хочу выборочно определить конструктор A:: A(int) только для некоторых типов с помощью enable_if. Для всех других типов существует конструктор по умолчанию A::A(...), которые должны быть по умолчанию для компилятора при замене не. Однако это имеет смысл для меня компилятор (gcc версии 4.9.0 20130714) по-прежнему жалуется

sfinae.cpp: в создании экземпляра 'struct A': sfinae.cpp: 19: 11:
требуется sfinae.cpp: 9: 5: ошибка: нет типа с именем "type" в
'struct std:: enable_if'
A ( int n): val(n) {};

возможно ли что-то подобное для конструктора? Возможно ли это с другим конструктором(конструкторами) (copy-constructor и move-constructor)?

2 ответов


Я думаю, что это не может работать с одним параметром шаблона по умолчанию, потому что его значение должно быть разрешено при создании экземпляра шаблона класса.

нам нужно отложить подстановку до момента создания экземпляра шаблона конструктора. Один из способов-установить по умолчанию параметр шаблона в T и добавить дополнительный фиктивный параметр в конструктор:

template<typename U = T>
A(int n, typename std::enable_if<U::value>::type* = 0) : val(n) { }

обычно это делается с помощью анонимного аргумента по умолчанию:

A(int n, typename std::enable_if<T::value>::type* = 0) : val(n) {};

вы не можете использовать параметры шаблона из класса В методы SFINAE out. Таким образом, один из способов-добавить фиктивный тип, заменяющий int:

см.:http://ideone.com/2Gnyzj

#include <iostream>
#include <type_traits>

template <typename T>
struct A {
    int val = 0;

    template<typename Integer
            ,typename  = typename std::enable_if<T::value && sizeof(Integer)>::type
            >
    A(Integer n) : val(n) {};

    A(...) {}
    /* ... */
};

struct YES { constexpr static bool value = true; };
struct NO { constexpr static bool value = false; };

int main() {
    A<YES> y(10);
    A<NO> n;
    std::cout << "YES: " << y.val << std::endl
              << "NO:  " << n.val << std::endl;
}

это работает, потому что вы используете параметр шаблона элемента для SFINAE из конструктора, но тест всегда истинен, поэтому он не загрязняет ваш чеки