Re: (what ought to be a) simple template question
On Apr 16, 5:25 am, Ian <ianm...@comcast.net> wrote:
Hi! Here is something that has been driving me crazy. Imagine the following
header and two cpp files:
// file.h
#include <stdio.h>
template<class T>
class X
{
public:
void nonInlineFunction() const;
};
template<class T>
void X<T>::nonInlineFunction() const
{
for( int i = 0; i < 2; i++ ); // here only to make it noninline
printf( "Generic\n" );
}
void f();
// file1.cpp
#include "file.h"
template<>
void X<int>::nonInlineFunction() const
{
for( int i = 0; i < 2; i++ ); // here only to make it noninline
printf( "Specialized for int\n" );
}
void f()
{
X<int> x;
x.nonInlineFunction();
}
// file2.cpp
#include "file.h"
int main()
{
X<int> x;
x.nonInlineFunction();
f();
return 0;
}
When I compile and run these, I get the following output
(a) g++ 2.81:
Generic
Specialized for int
(b) g++ 2.91.66
Specialized for int
Specialized for int
(c) Visual C++
Does not link, the error is
file1.obj : error LNK2005: "public: void __thiscall
X<int>::nonInlineFunction(void)const " (?nonInlineFunction@?$X@H@@QBEXXZ)
already defined in file2.obj
What is going on here?
One Definition Rule (ODR) is violated here, which causes undefined
behaviour.
What I want to accomplish is a specialized version of the nonInlineFunction()
for int, where the definition cannot be inline. This should be pretty simple,
no? The behavior I want is as displayed in case (b) above. Based on the books I
have read (e.g. "C++ Templates The Complete Guide") I would expect to see case
(a), because when compiling file2.cpp, the compiler knows nothing about the
specialization in file1.cpp and should thus instantiate from the generic
template.
This is all good, however, it is when you link these translation units
into one program you break the ODR.
I didn't know how to fix this, because if I include the full
definition of the specialized function in the header, I get multiple difinitions
(one for each file).
This is expected, because a (full) function template specialisation is
not a function template any more, rather it is a regular function.
Thus, if you provide its definition in a header it should be inline.
I thought that adding the line
template<> void X<int>::nonInlineFunction() const;
in the header would fix this by telling the compiler that there is a
specialization of the function for "int",
I would expect something like this should fix that.
but it seems to do nothing.
Does it not change in any way the behaviour you are observing?
So:
1) What is the right way of providing a specialized version of the function that
is not inline?
By providing a declaration in a header and the definition in one of
the translation units that comprise your application.
2) Do you have any suggestions how to coerce VC++ into compiling this?
May be try adding extern to the declaration of the specialisation in
the header (don't have M$VC to check):
extern template<> void X<int>::nonInlineFunction() const;
--
Max
--
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]