Note: this seems to be a repost of a problem: C++ – Overload templated class method with a partial specilization of that method
I have boiled down a problem I am having with C++ template specialization down to a simple case.
It consists of a simple 2-parameter template class Thing, where I would like to specialize Thing<A,B>::doSomething() for B=int.
#include <cstdio>
// A 3-parameter template class.
template <class A, class B>
class Thing
{
public:
Thing(A a, B b) : a_(a), b_(b) {}
B doSomething();
private:
A a_;
B b_;
};
// The generic case works as expected.
template <class A, class B>
B Thing<A,B>::doSomething()
{
return b_;
}
// This specialization does not work!
template <class A>
int Thing<A,int>::doSomething()
{
return b_+1;
}
int main() {
// Setup our thing.
Thing<double,int> thing(1.0,2);
// This doesn't compile - but works with the generic case.
printf("Expecting 3, and getting %i\n", thing.doSomething());
// Clean up.
return 0;
}
Unfortunately, g++ exits with the error:
partial_specialization.cpp:30: error: invalid use of incomplete type ‘class Thing<A, int>’
partial_specialization.cpp:8: error: declaration of ‘class Thing<A, int>’
The clang++ compiler is a bit more verbose, but has the same problem:
partial_specialization.cpp:30:19: error: nested name specifier 'Thing<A, int>::' for declaration does not
refer into a class, class template or class template partial specialization
int Thing<A,int>::doSomething()
~~~~~~~~~~~~~~^
partial_specialization.cpp:32:12: error: use of undeclared identifier 'b_'
return b_+1;
^
2 errors generated.
I have read and understood that partial template specializations on functions aren’t allowed – but I thought I was partially specializing over classes of Thing in this case.
Any ideas?
What I did: A workaround, as determined from the link provided by the accepted answer:
template< class T >
inline T foo( T const & v ) { return v; }
template<>
inline int foo( int const & v ) { return v+1; }
// The generic case works as expected.
template <class A, class B>
B Thing<A,B>::doSomething()
{
return foo(b_);
}
Partial specialization of a function template, whether it is member function template or stand-alone function template, is not allowed by the Standard:
But you can partially specialize the class template itself. You can do something like this:
Note that when you partially specialize a class template, then the template parameter-list of member function (in its definition outside the class), must match the template parameter list of the class template partial specialization. That means, for the above partial specialization of the class template, you cannot define this:
Its not allowed, because the template parameter-list in function definition didn’t match the template parameter-list of the class template partial specialization. §14.5.4.3/1 from the Standard (2003) says,
For more on this, read my answer here:
C++ – Overload templated class method with a partial specilization of that method
So what is the solution? Would you partially specialize your class along with all the repetitive work?
A simple solution would be work delegation, instead of partially specializing the class template. Write a stand-alone function template and specialize this as:
And then call this function template from
doSomething()member function as:Since in your particular case,
doTheActualSomethingneeds to know the value of only one member, namelyb_, the above solution is fine, as you can pass the value to the function as argument whose type is the template type argumentB, and specialization forintis possible being it full-specialization.But imagine if it needs to access multiple members, type of each depends on the template type argument-list, then defining a stand-alone function template wouldn’t solve the problem, because now there will be more than one type argument to the function template, and you cannot partially specialize the function for just, say, one type (as its not allowed).
So in this case you can define a class template instead, which defines a static non-template member function
doTheActualSomething. Here is how:Notice that you can use
thingpointer to access any member of the class. Of course, if it needs to access private members, then you’ve to makestruct Workera friend ofThingclass template, as:Now delegate the work to the friend as:
Two points to be noted here:
doTheActualSomethingis not a member function template. Its not enclosing class which is template. Hence we can partially specialize the class template anytime, to get the desired effect of the partial member function template specialization.thispointer as argument to the function, we can access any member of the classThing<A,B>, even private members, asWorker<T,U>is also a friend.Complete online demo : http://www.ideone.com/uEQ4S
Now there is still a chance of improvement. Now all instantiations of
Workerclass template are friends of all instantiation ofThingclass template. So we can restrict this many-to-many friendship as:Now only one instantiation of
Workerclass template is a friend of one instantiation ofThingclass template. That is one-to-one friendship. That is,Worker<A,B>is a friend ofThing<A,B>.Worker<A,B>is NOT a friend ofThing<A,C>.This change requires us to write the code in somewhat different order. See the complete demo, with all the ordering of class and function definitions and all:
http://www.ideone.com/6a1Ih