走进C++11(十六)外部模板

 

关键词

 

extern

 

语法

 

extern template class|struct 模板名 < 实参列表 > ;    

 

解释

 

类模板自身并不是类型、对象或任何其他实体。不会从从仅含模板定义的源文件生成任何代码。必须实例化模板以令任何代码出现:必须提供模板实参,使得编译器能生成实际的类(或从函数模板生成函数)。

 

如果外部模板声明出现于某个编译单元中,那么与之对应的显式实例化必须出现于另一个编译单元中或者同一个编译单元的后续代码中;


外部模板不能用于一个静态函数(没有外部链接属性),但可以用于类静态成员函数。

 

类模板实例化分为两种:显示实例化和隐式实例化。

 

显示实例化有如下方法:

 

template class|struct 模板名 < 实参列表 > ;

extern template class|struct 模板名 < 实参列表 > (C++11 起);

 

第二种方法就是我们要说的C++11新增的方法。

 

隐式实例化:

 

当代码在要求完整定义的类型的语境中涉指某个模板时,或当类型的完整性对代码有影响,而这个特定类型尚未被显式实例化时,发生隐式实例化。例如当构造此类型的对象之时,但不包括构造指向此类型的指针之时。举个例子:

 

template<class T> struct Z {    void f() {}    void g(); // 并不定义}; // 模板定义template struct Z<double>; // 显式实例化 Z<double>Z<int> a; // 隐式实例化 Z<int>Z<char>* p; // 此处不实例化任何内容p->f(); // 隐式实例化 Z<char> 而 Z<char>::f() 出现于此。// 并不需要且始终不实例化 Z<char>::g():不必对其进行定义

 

WHY

 

而对于函数模板来说,现在我们遇到的问题和extern一个变量遇到的问题相同。不同的是,发生问题的不是变量(数据),而是函数(代码)。这样的困境是由于模板的实例化带来的。

 

比如,我们在一个test.h的文件中声明了如下一个模板函数:

 

template <typename T> void fun(T) {}

 

在第一个test1.cpp文件中,我们定义了以下代码:

 

#include "test.h"void test1() { fun(3); }

 

而在另一个test2.cpp文件中,我们定义了以下代码:

 

#include "test.h"void test2() { fun(4); }

 

由于两个源代码使用的模板函数的参数类型一致,所以在编译test1.cpp的时候,编译器实例化出了函数 fun(int),而当编译test2.cpp的时候,编译器又再一次实例化出了函数fun(int)。那么可以想象,在test1.o目标文件和test2.o目标文件中,会有两份一模一样的函数fun(int)代码。

 

代码重复和数据重复不同。数据重复,编译器往往无法分辨是否是要共享的数据;而代码重复,为了节省空间,保留其中之一就可以了(只要代码完全相同)。事实上,大部分链接器也是这样做的。在链接的时候,链接器通过一些编译器辅助的手段将重复的模板函数代码fun(int)删除掉,只保留了单个副本。这样一来,就解决了模板实例化时产生的代码冗余问题。

 

不过读者也注意到了,对于源代码中出现的每一处模板实例化,编译器都需要去做实例化的工作;而在链接时,链接器还需要移除重复的实例化代码。很明显,这样的工作太过冗余,而在广泛使用模板的项目中,由于编译器会产生大量冗余代码,会极大地增加编译器的编译时间和链接时间。解决这个问题的方法基本跟变量共享的思路是一样的,就是使用“外部的”模板。

 

C++11我们可以通过下面代码来实现显示实例化:

 

extern template void fun<int>(int);

 

这样一来,在test2.o中不会再生成fun(int)的实例代码。由于test2.o不再包含fun(int)的实例,因此链接器的工作很轻松,基本跟外部变量的做法是一样的,即只需要保证让test1.cpp和test2.cpp共享一份代码位置即可。而同时,编译器也不用每次都产生一份fun(int)的代码,所以可以减少编译时间。这里也可以把外部模板声明放在头文件中,这样所有包含test.h的头文件就可以共享这个外部模板声明了。这一点跟使用外部变量声明是完全一致的。

 

关注公众号获取更多信息:

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值