Re: Specialization oddities
Alf P. Steinbach schrieb:
* Daniel Kr?gler:
The point I want to discuss is: How come that we
have more restrictions here for explicit specializations
compared to partial specialization?
First, for other readers: the standard's "explicit specialization"
really means "full specialization", that all template parameters have
been fixed to actual types or values.
Although this term does not exist in the standard,
it is often used in this sense, yes.
Consider now
template< typename T >
struct Argh
{
template< typename U > struct Bah {};
};
template< typename T >
template<> // Not allowed.
struct Argh<T>::Bah<float> { T ahum; };
Well, there's no way that machine code can be generated for Bah<T,
float>, because it's not fully specialized (not considering here the
kind of code that would be generated for "export").
That is correct. Personally I never have seen at an
explicit specialization in the way, you present it
here, I always have seen the result of a full specialization
as a simpler form of specialization as e.g. partial
specialization - which probably was an error.
The standard seems to strengthen you interpretation,
if I read 14.7/4 (N2134) carefully:
"[..] A specialization is a class, function, or class member
that is either instantiated or explicitly specialized (14.7.3)."
The fun part is, that a partial specialization is not a
specialization by this definition ;-)
So, my hypothesis goes, instead of allowing that, the standard landed on
giving "template<>" the clear, unambigious meaning of "fully
specialized" (which it calls explicit specialization), meaning that the
class or function can be instantiated, that machine code can be generated.
Your answer gave me a complete different angle to look
at this situation - Thank you!
And if something like the above is desired, then it can always be
rewritten as partial specialization, like
template< typename T, typename U > struct ArghBah {};
template< typename T >
struct Argh
{
template< typename U >
struct Bah { typedef ArghBah<T, U> Type; };
};
template< typename T >
struct ArghBah<T, float> { T ahum; };
It's just a little notational inconvenience, writing
"Argh<bool>::Bah<float>::Type" instead of "Argh<bool>::Bah<float>".
Yes, it is possible to find workarounds - fortunately. But
it would not be a "natural solution", which could be applied
for a simple customization point, e.g. as James Kanze
recently proposed to add a member operator template
in basic_i/ostream to allow user-extended inserters and
extractors. Of course one could choose an alternative
route, which uses free operator function templates
(which might be even better).
I speculated for a minute or so that without this rule one could risk
exponential or infinite code generation, but then I failed to come up
with any example, so I don't think it's that, just clear notation.
I also think that this is probably not the reason,
because there cannot be any increased complexity
compared to partial specialization and this specialization
is in the same not automatic instantiated, because it
is still bound to not yet fixed template parameters.
Thanks for your valuable opinon,
Daniel
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]