使用extern模板(C ++ 11)圖1:功能模板TemplHeader.htemplate<typename T>void f();TemplCpp.cpptemplate<typename T>void f(){
//...} //explicit instantationtemplate void f<T>();Main.cpp的#include "TemplHeader.h"extern template void f<T>(); //is this correct?int main() {
f<char>();
return 0;}這是正確的使用方法extern template,還是僅將此關鍵字用于類模板,如圖2所示?圖2:類模板TemplHeader.htemplate<typename T>class foo {
T f();};TemplCpp.cpptemplate<typename T>void foo<T>::f() {
//...}//explicit instantationtemplate class foo<int>;Main.cpp的#include "TemplHeader.h"extern template class foo<int>();int main() {
foo<int> test;
return 0;}我知道將所有這些放在一個頭文件中是好的,但如果我們在多個文件中實例化具有相同參數的模板,那么我們會得到多個相同的定義,編譯器會將它們全部刪除(除了一個)以避免錯誤。我該怎么用extern template?我們可以只將它用于類,還是可以將它用于函數?此外,圖1和圖2可以擴展為模板位于單個頭文件中的解決方案。在這種情況下,我們需要使用extern template關鍵字來避免多個相同的瞬時。這僅適用于課程或功能嗎?
3 回答

12345678_0001
TA貢獻1802條經驗 獲得超5個贊
當您知道它將在其他地方實例化時,您應該只使用extern template
強制編譯器不實例化模板。它用于減少編譯時間和目標文件大小。
例如:
// header.htemplate<typename T>void ReallyBigFunction(){ // Body}// source1.cpp#include "header.h"void something1(){ ReallyBigFunction<int>();}// source2.cpp#include "header.h"void something2(){ ReallyBigFunction<int>();}
這將導致以下目標文件:
source1.o void something1() void ReallyBigFunction<int>() // Compiled first timesource2.o void something2() void ReallyBigFunction<int>() // Compiled second time
如果兩個文件都鏈接在一起,void ReallyBigFunction<int>()
則會丟棄一個文件,從而導致浪費的編譯時間和目標文件大小。
為了不浪費編譯時間和目標文件大小,有一個extern
關鍵字使編譯器無法編譯模板函數。當且僅當您知道它在其他地方使用相同的二進制文件時,您應該使用它。
更改source2.cpp
到:
// source2.cpp#include "header.h"extern template void ReallyBigFunction<int>();void something2(){ ReallyBigFunction<int>();}
將導致以下目標文件:
source1.o void something1() void ReallyBigFunction<int>() // compiled just one timesource2.o void something2() // No ReallyBigFunction<int> here because of the extern
當這兩者都鏈接在一起時,第二個目標文件將只使用第一個目標文件中的符號。不需要丟棄,也不需要浪費編譯時間和目標文件大小。
這應該僅在項目中使用,例如在多次使用模板時vector<int>
,您應該extern
在除一個源文件之外的所有文件中使用。
這也適用于類和函數作為一個,甚至模板成員函數。

陪伴而非守候
TA貢獻1757條經驗 獲得超8個贊
模板的已知問題是代碼膨脹,這是在調用類模板特化的每個模塊中生成類定義的結果。為了防止這種情況,從C ++ 0x開始,可以在類模板特化之前使用關鍵字extern
#include <MyClass>
extern template class CMyClass<int>;
模板類的顯式實例應僅在單個轉換單元中發生,最好是具有模板定義的轉換單元(MyClass.cpp)
template class CMyClass<int>;template class CMyClass<float>;
- 3 回答
- 0 關注
- 1299 瀏覽
添加回答
舉報
0/150
提交
取消