So I have a fairly complex function:
template <typename T>
void foo(const int param1, const int param2, int& out_param)
Given int bar
, const int arg1
, and const int arg2
the function will be called with either: foo<plus<int>>(arg1, arg2, bar)
or foo<minus<int>>(arg1, arg2, bar)
Internally the function is rather complex but I am doing different relational operators based on the type of functor that was passed as a template parameter.
In the case of plus
I need to do:
arg1 > arg2
bar > 0
bar > -10
In the case of minus
I need to do:
arg1 < arg2
bar < 0
bar < 10
Note that 10
does not have the same sign in both 3s. I am currently solving all this by passing a second template parameter (less
or greater
.) But I was thinking it might make more sense to write these relations as arithmetic operations. Is that even possible, or do I need to take the second template parameter?
The basic idea is
a > b
if and only if-a < -b
. Andplus(0,a)==a
whileminus(0,a)==-a
.The last one is tricky, as we want to change the order of
<
and the sign. Luckily they cancel:Suppose we want a constant that is
-10
in the plus case, and10
in the minus case. Thenis
-10
andis
10
.So we get:
in the plus case, the rhs is
0+0+-10
, aka-10
.In the minus case, this is
0-(0-(-10))
, aka-10
.So the short form is:
and it should work.
Besides @Yakk's answer there are a number of ways you can do this. Here are 5.
Method 1: Function traits
This is more of a classic technique used before more advanced template-metaprogramming techniques became available. It's still quite handy. We specialize some structure depending on
T
to give us the types and constants we want to use.Live Demo 1
Method 2: Full specialization
Another "classic" technique that remains useful. You are probably familiar with this technique, but I'm showing it for completeness. So long as you never need to partially specialize a function, you can write different version of it for the types you need:
Live Demo 2
Method 3: Tagged dispatch
A third classic technique that turns a type check into an overloading problem. The gist is that we define some lightweight
tag
struct that we can instantiate, and then use that as a differentiator between overloads. Often this is nice to use when you have a templated class function, and you don't want to specialize the entire class just to specialize said function.Live Demo 3
Method 4: Straightforward
constexpr if
Since C++17 we can use
if constexpr
blocks to make a compile-time check on a type. These are useful because if the check fails the compiler doesn't compile that block at all. This oftentimes leads to much easier code than before, where we had to use complicated indirection to classes or functions with advanced metaprogramming:Live Demo 4
Method 5:
constexpr
+ trampoliningtrampolining is a metaprogramming technique where you use a "trampoline" function as an intermediary between the caller and the actual function you wish to dispatch to. Here we will use it to map to the appropriate comparison type (
std::greater
orstd::less
) as well as the integral constants we wish to comparebar
to. It's a little more flexible than Method 4. It also separates concerns a bit, too. At the cost of readability:Live Demo 5