Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Template definitions outside class body

Tags:

Is it O.K. to define virtual function of class template outside its body? Virtual function can not be inlined, but to avoid multiple definitions in compilation units they shall be marked inline (assuming that template headers will be included in multiple source files). On the other hand compiler is free to ignore inline, so this seems valid. By an example, is the code below correct:

template <typename T>
class C
{
public:
    virtual void f(T val);
};

template <typename T>
inline
void C<T>::f(T val)
{
  //definition
}

?

BTW gcc (3.4.2) allows to omit inline before definition of function f(T val) but not before analogous function of regular class. Is it only gcc's behaviour?

like image 337
mip Avatar asked Apr 18 '12 14:04

mip


2 Answers

Yes, it's OK even without inline. It works the same for ordinary member functions and static variables:

// everything in the header:
template <class T>
class A
{
  static int i;
};

template <class T>
int A<T>::i=0;

Standard quote: (3.2/5)

There can be more than one definition of a class type (Clause 9), enumeration type (7.2), inline function with external linkage (7.1.2), class template (Clause 14), non-static function template (14.5.6), static data member of a class template (14.5.1.3), member function of a class template (14.5.1.1), or template specialization for which some template parameters are not specified (14.7, 14.5.5) in a program provided that each definition appears in a different translation unit, and provided the definitions satisfy the following requirements ...

The requirements basically say the two definitions have to be identical.

It doesn't work in case of regular classes. There has to be at most one definition in the whole program.

like image 189
jpalecek Avatar answered Oct 08 '22 08:10

jpalecek


You can define template methods outside the class definition, in the same header, without using inline and without receiving multiple definition errors.

That's because a template function doesn't generate a definition itself, if it's not fully specialized. To prove my point, the following:

void C<int>::f(int)
{
}

will result in a linker error, as the function has a definition in this case. (provided you include this in multiple translation units. If you mark it inline:

inline void C<int>::f(int)
{
}

the error no longer occurs.

like image 26
Luchian Grigore Avatar answered Oct 08 '22 06:10

Luchian Grigore