Re: Can static member variables be declared "inline"?
On Sep 2, 12:26 am, Juha Nieminen <nos...@thanks.invalid> wrote:
As we know, the keyword "inline" is a bit misleading because
its meaning has changed in practice. In most modern compilers
it has completely lost its meaning of "a hint for the compiler
to inline the function if possible" (because if the compiler
has the function definition available at an instantiation
point, it will estimate whether to inline it or not, and do so
if it estimates it would be beneficial, completely regardless
of whether the keyword 'inline' appears in the function
definition or not).
Unless the compiler can be reasonably sure of making a better
estimate than you, I'd call this a bug, or at least poor quality
of implementation. And only a very few compilers are capable of
this except in very limited cases.
In fact, the keyword "inline" is in practice a linker command,
not a compiler command.
It's not a command, it's a "function specifier", which is in
some ways a bit like a "storage class specifier". It's a part
of the language, and has a specific semantic associated with the
language (it makes multiple definitions legal, provided that
they are in different sources). It also has an "intent", and it
is a poor compiler which ignores this intent (unless, as I said,
it is certain that it can do better).
It tells the linker that if it finds an instantiation of that
function in more than one object file, it should use only one
of them and discard the rest.
No. That's one aspect of some implementations, but it's not
fundamental. (And a compiler could do this with non-inline
functions as well, if it wanted to.)
Also as we know, template functions and member functions of
template classes are implicitly "inline":
No they're not, although they can also be multiply defined (and
you're forgetting export, of course).
If they are instantiated in more than one compilation unit
(and not inlined), the linker will use only one instantiation
when linking and discard the rest.
Again, implementation defined. I've used compilers that didn't
instantiate any of the template functions until link time, and
then only once.
What many C++ programmers might not realize is that this
implicit inlining of template class members extends to static
member variables as well. In other words, if you do this:
// Template class header file
// --------------------------------------------
template<typename T>
class SomeClass
{
static int memberVariable;
...
};
template<typename T>
int SomeClass<T>::memberVariable = 1;
// --------------------------------------------
and this header is used in more than one compilation unit, the
static member variable will also be instantiated in all those
compilation units, but this will not cause a linker error.
Again, that's one way (not the best, but probably the most
widespread) of implementing this. It's certainly not required
by the standard.
The linker will nicely use only one of them and discard the
rest. Effectively the static member variable has been
declared "inline".
This is not only a nice feature of template classes, but a
must. Without this property it would be impossible to have
static member variables in template classes because it would
be impossible to instantiate them in only one place (if the
template class is used in more than one compilation unit).
So my question is: Why can't static member variables of
regular classes be declared "inline", like the ones of
template classes?
Because "the inline specifier indicates to the implementation
that inline substitution of the function body at the point of
call is to be preferred to the usual function call mechanism."
Which obviously doesn't make sense for variables. And "while an
implementation is not required to perform this inline
substitution", the intent is clearly expressed, and it would be
a very poor implementation which didn't perform it without a
very good reason for not doing so.
Compilers clearly have the support for this because they must
do it for template classes. However, seemingly the C++
standard does not support this for static members of regular
classes. Or is there a way that I don't know of?
Make the class a template:-).
Seriously, why do you want to do it?
--
James Kanze (GABI Software) email:james.kanze@gmail.com
Conseils en informatique orient=E9e objet/
Beratung in objektorientierter Datenverarbeitung
9 place S=E9mard, 78210 St.-Cyr-l'=C9cole, France, +33 (0)1 30 23 00 34