2015-12-10 17 views
5

To jest moja pierwsza próba SFINAE:Nie można uzyskać SFINAE pracować

#include <type_traits> 
#include <iostream> 

struct C1 { 
    using T = int; 
}; 

struct C2 { 
    using T = void; 
}; 

// For classes that declare T = int 
template <class C> 
void f(C &c, 
     std::enable_if<!std::is_same<typename C::T, void>::value, int>::type = 0) { 
    std::cout << "With T" << std::endl; 
} 

// For classes that declare T = void 
template <class C> 
void f(C &c, 
     std::enable_if<std::is_same<typename C::T, void>::value, int>::type = 0) { 
    std::cout << "Without T" << std::endl; 
} 

int main() { 
    C1 c1; 
    f(c1); // With T 
    C2 c2; 
    f(c2); // Without T 
    return 0; 
} 

kompilator (gcc 4.8.2) narzeka:

‘std::enable_if<!(std::is_same<typename C::T, void>::value), int>::type’ is not a type 

Co robię źle?

+2

Musisz napisać 'TypeName std :: enable_if <...> :: type' ponieważ 'type' jest zależny. [Przeczytaj to] (https://stackoverflow.com/questions/610245/where-and-why-do-i-have-to-put-the-template-and-typename-keywords). Jeśli możesz użyć C++ 14, możesz zamiast tego użyć 'std :: enable_if_t <...>' i nie będziesz miał tego problemu. – Simple

+0

Najlepiej byłoby umieścić 'enable_if's w ​​parametrach szablonu, aby nie zaszumieć argumentów funkcji –

+0

Zrobiłem edycję twojego kodu. Zakładam, że były to rzeczy, które po prostu pominąłeś lub nieprawidłowo wpisałeś, ale dodałem je dla jasności. – erip

Odpowiedz

7

Trzeba parę typename s dla tej pracy:

// For classes that declare T = int 
template <class C> 
void f(C &c, 
     typename std::enable_if<!std::is_same<typename C::T, void>::value, int>::type = 0) { 
    std::cout << "With T" << std::endl; 
} 

// For classes that declare T = void 
template <class C> 
void f(C &c, 
     typename std::enable_if<std::is_same<typename C::T, void>::value, int>::type = 0) { 
    std::cout << "Without T" << std::endl; 
} 

lub jeśli kompilator C++ obsługuje 14 można użyć std::enable_if_t:

// For classes that declare T = int 
template <class C> 
void f(C &c, 
     std::enable_if_t<!std::is_same<typename C::T, void>::value, int> = 0) { 
    std::cout << "With T" << std::endl; 
} 

// For classes that declare T = void 
template <class C> 
void f(C &c, 
     std::enable_if_t<std::is_same<typename C::T, void>::value, int> = 0) { 
    std::cout << "Without T" << std::endl; 
} 
Powiązane problemy