Re: templated class : constructor specialization
JSeb ha scritto:
Good afternoon, gurus and others!
I have a template class that looks like this:
template <typename T>
class CTemplateClass
{
private:
const T* const mpData;
public:
CTemplateClass<T>(const T& aData) : mpData(new T(aData)) { }
virtual ~CTemplateClass<T>()
{
delete this->mpData;
}
};
You don't need to use CTemplateClass<T> inside the template body.
CTemplateClass will just do.
I want to write a SPECIALIZED CONSTRUCTOR to deal with pointer types.
From what I found online, I should write something like that AFTER the
code above:
template <typename T>
class CTemplateClass<T*>
{
public:
CTemplateClass<T*>(const T* apData) : mpData(apData) { }
Again: CTemplateClass will do, here. Don't be fooled by the
CTemplateClass<T*> after "class" above, which instead is required.
};
But :^( the compiler tells me that CTemplateClass<T*> has no field
named "mpData".
That's correct: you didn't provide a declaration for it and you should
have done it.
I'm not too familiar with templates, and I don't quite understand the
syntax involved in the specialization. To me, it looks as if
"CTemplateClass" is defined twice, with the second definition indeed
lacking the attribute "mpData".
A specialization is a completely new definition, which does not have
access to the "main" definition. The only relation between a template
and one of its specialization is the name. Nothing else. This may seems
strange, but, think about it... how could it be otherwise? How could the
compiler decide that the line:
CTemplateClass<T*>(const T* apData) : mpData(apData) { }
in the specialization is meant to be a replacement of
CTemplateClass<T>(const T& aData) : mpData(new T(aData)) { }
in the main template? The two functions have completely different and
unrelated signatures! If you were right and the specialization had
access to the main template, then you would end up with two
constructors, namely:
CTemplateClass(T* const& aData) : mpData(new T(aData)) { }
CTemplateClass(const T* apData) : mpData(apData) { }
Notice that the first one takes a const reference to a pointer to T,
which is completely different from the second, which takes a pointer to
const T. (Be careful about where the const goes!)
Requiring the programmer to provide a full definition gives you full
control on the specialization, in the sense that not only you can add or
replace features, you can also remove them.
And last but not least, you are even free to omit the main definition!
For example, this idiom is perfectly valid and can sometime be useful:
template <class T>
class MyTemplate; // declared but not defined
template <class T>
class MyTemplate<T*>
{
// definition here
};
I don't get it : if "CTemplateClass<T*>" is a specialization. doesn't
it have access to "CTemplateClass<T>" attributes and methods? (I know
the answer is "no", but I fail to see why. In fact, from what I've
read, both templates don't even need to share any interface).
The answer is no because CTemplateClass<T> and CTemplateClass<T*>,
except for (part of) the name, they are totally unrelated types.
By the way, if I define "mpData" in the specialization as in the
original template, it works.
Sure it does.
HTH,
Ganesh
PS: exercise for the reader: give the OP's definitions, what is the
destructor of CTemplateClass<T*>?
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]