Here is an example case of what I’m trying to do (it is a “test” case just to illustrate the problem) :
#include <iostream>
#include <type_traits>
#include <ratio>
template<int Int, typename Type>
constexpr Type f(const Type x)
{
return Int*x;
}
template<class Ratio, typename Type,
class = typename std::enable_if<Ratio::den != 0>::type>
constexpr Type f(const Type x)
{
return (x*Ratio::num)/Ratio::den;
}
template</*An int OR a type*/ Something, typename Type>
constexpr Type g(const Type x)
{
return f<Something, Type>(x);
}
int main()
{
std::cout<<f<1>(42.)<<std::endl;
std::cout<<f<std::kilo>(42.)<<std::endl;
}
As you can see, there are two versions of the f() function : the first one takes an int as a template parameter, and the second one takes a std::ratio. The problem is the following :
I would like to “wrap” this function through g() which can take an int OR a std::ratio as first template parameter and call the good version of f().
How to do that without writing two g() functions ? In other words, what do I have to write instead of /*An int OR a type*/ ?
Here’s how I would do it, but I’ve changed your interface slightly:
Notes:
I’ve taken advantage of
constexprto not pass compile-time constants via template parameters (that’s whatconstexpris for).gis now just a perfect forwarder. However I was unable to usestd::forwardbecause it isn’t marked up withconstexpr(arguably a defect in C++11). So I dropped down to usestatic_cast<T&&>instead. Perfect forwarding is a little bit overkill here. But it is a good idiom to be thoroughly familiar with.