Search code examples
c++c++11templatessfinaeenable-if

Selecting a member function using different enable_if conditions


I am trying to determine which version of a member function gets called based on the class template parameter. I have tried this:

#include <iostream>
#include <type_traits>

template<typename T>
struct Point
{
  void MyFunction(typename std::enable_if<std::is_same<T, int>::value, T >::type* = 0)
  {
    std::cout << "T is int." << std::endl;
  }

  void MyFunction(typename std::enable_if<!std::is_same<T, int>::value, float >::type* = 0)
  {
    std::cout << "T is not int." << std::endl;
  }
};

int main()
{
  Point<int> intPoint;
  intPoint.MyFunction();

  Point<float> floatPoint;
  floatPoint.MyFunction();
}

which I thought is saying "use the first MyFunction if T is int, and use the second MyFunction if T is not int, but I get compiler errors saying "error: no type named ‘type’ in ‘struct std::enable_if’". Can anyone point out what I am doing wrong here?


Solution

  • enable_if works because the substitution of a template argument resulted in an error, and so that substitution is dropped from the overload resolution set and only other viable overloads are considered by the compiler.

    In your example, there is no substitution occurring when instantiating the member functions because the template argument T is already known at that time. The simplest way to achieve what you're attempting is to create a dummy template argument that is defaulted to T and use that to perform SFINAE.

    template<typename T>
    struct Point
    {
      template<typename U = T>
      typename std::enable_if<std::is_same<U, int>::value>::type
        MyFunction()
      {
        std::cout << "T is int." << std::endl;
      }
    
      template<typename U = T>
      typename std::enable_if<std::is_same<U, float>::value>::type
        MyFunction()
      {
        std::cout << "T is not int." << std::endl;
      }
    };
    

    Edit:

    As HostileFork mentions in the comments, the original example leaves the possibility of the user explicitly specifying template arguments for the member functions and getting an incorrect result. The following should prevent explicit specializations of the member functions from compiling.

    template<typename T>
    struct Point
    {
      template<typename... Dummy, typename U = T>
      typename std::enable_if<std::is_same<U, int>::value>::type
        MyFunction()
      {
        static_assert(sizeof...(Dummy)==0, "Do not specify template arguments!");
        std::cout << "T is int." << std::endl;
      }
    
      template<typename... Dummy, typename U = T>
      typename std::enable_if<std::is_same<U, float>::value>::type
        MyFunction()
      {
        static_assert(sizeof...(Dummy)==0, "Do not specify template arguments!");
        std::cout << "T is not int." << std::endl;
      }
    };