< >中谈到过,模板的文件组织一节中就讲到了这个问题. 书中提到了2中方法:
第一种:实现和声明放在一个编译单位中,这样这个模板定义所依靠的一些东西就被带到了包含这个编译单位的文件里。
坏处是:增大编译器的处理信息量,而且用户会意外的依赖了原本只是为定义这个模板而引进的东西。
解决方案:使用名字空间、避免宏定义、以及一般性的减少包含进来的信息量,可以减小这种危害影响。
第二种:分开编译。这种处理就和一般的非inline函数处理一样。不过这里需要用到export关键字了,export关键字的作用就是告诉编译器这个东西别的编译单位可以访问;否则按照一般规则,任何地方使用模板,都要求其定义在作用域中。
这种方法的优点是:思路清晰。
缺点:因为模板被单独编译了,因此在需要时候,找到合适的定义成了实现者(编译器的责任),要实现者在需要时候找到唯一的一个版本,想想编译器这家伙可就头疼了,因此很少编译器做到这一点。
总结了上面的两种情况后,我们可以看到,第一种在编译时候,任何需要这个模板的地方都引用了这个模板定义,因此编译器据此来优化,去掉冗余的版本。
第二种方法就给实现者(编译器)增加了负担,不是去掉冗余的版本,而是在需要时找到这个唯一版本。 本人也在自己的VS2005上测试,出现提示 C/C++ code.convert.cpp(6) : warning C4237: 目前还不支持“export”关键字,但已保留该关键字供将来使用
filename: seqlist.h 类模板的申明
////////////////////////////////////////////////////////////
#ifndef SEQLIST
#define SEQLIST
template<calss T>
class SeqList
{
}
////////////////////////////////////////////////////////////
// filename: seqlist.cpp 类模板的定义
////////////////////////////////////////////////////////////
#include"seqlist.h"
template<class T>
SeqList<T>::SeqList()
{
}
////////////////////////////////////////////////////////////
// filename: main.cpp 主函数
////////////////////////////////////////////////////////////
#include"seqlist.h"
#include<iostream>
int main()
{
}
如以上写法,连接时会出错,这个问题属于类模板的分离编译,有三种解决办法:
1. 将类模板申明与类模板定义放到一个文件中。
2. 在主函数中加 #include"seqlist.cpp"。
3. 使用export关键字。
c++primer中讲到"c++的模板编译模式"有两种——"包含编译和分离编译"。详见c++primer 16.8
///////////////////////////////////////////////////////////////
在讨论分离编译之前,首先来看一下包含编译:
一、包含编译模式
在包含编译模式下,我们在每个模板被实例化的文件中包含函数模板的定义,并且往往把定义放在头文件中,象内联函数所做的那样。如:
// model1.h
// 包含模式:模板定义放在头文件中
template <typename Type>
Type min( Type t1, Type t2 ) {
return t1 < t2 ? t1 : t2;
}
在每个使用min()实例的文件中都包含了该头文件,如:
// 在使用模板实例之前包含模板定义
#include "model1.h"
int i, j;
double dobj = min( i, j );
该头文件可以被包含在许多程序文本文件中。这意味着编译器必须在每个调用该实例的文件中实例化min()的整型实例吗?不。该程序必须表现得好像min()的整型实例只被实例化一次。但是,真正的实例化动作发生在何时何地,要取决于具体的编译器实现。
二、分离编译模式
在分离编译模式下,函数模板的声明被放在头文件中。在这种模式下,函数模板声明和定义的组织方式与程序中的非内联函数的声明和定义组织方式相同。如:
// model2.h
// 分离模式:只提供模板声明
template <typename Type> Type min( Type t1, Type t2 );
// model2.c
// 模板定义
export template <typename Type>
Type min( Type t1, Type t2 ) {/* . . . */}
使用函数模板min()实例的程序只需在使用该实例之前包含这个头文件:
// user.c
#include "model2.h"
int i, j;
double d = min( i, j ); // OK: 用法,需要一个实例
我们通过在模板定义中的关键字template之前加上关键字export,来声明一个可导出的函数模板。
三、显式实例化声明
标准C++提供了显式实例化声明来帮助程序员控制模板实例化发生的时间。在显式实例化声明中,关键字template后面是函数模板实例的声明,其中显式地指定了模板实参。下例中提供了sum(int* , int)的显式实例化声明:
template <typename Type>
Type sum( Type op1, int op2 ) {/* . . . */} // 函数模板sum的定义必须给出
// 显式实例化声明
template int* sum< int* >( int*, int );
该显式实例化声明要求用模板实参int*实例化模板sum()。对于给定的函数模板实例,显式实例化声明在一个程序中只能出现一次。