您好,登錄后才能下訂單哦!
之前遇到了一些關(guān)于模板特例化的問題,自己總結(jié)一下。
模板的特例化是C++新標(biāo)準的一個特點,可以自定義某些模板的實現(xiàn),比如在比較函數(shù)compare可以使用less<T>標(biāo)準庫模板比較string、int、char、指針等類型,但如果有const char*類型且比較字符串的字典大小時,就與之前的比較方式不同了:
#ifndef A_H #define A_H #include <iostream> #include <cstring> using std::less; template <typename T> int compare(const T v1,const T v2) { if (less<T>()(v1,v2)) { return -1; } else if (less<T>()(v2,v1)) { return 1; } else { return 0; } } template <> inline int compare(const char *const a,const char *const b) { return strcmp(a,b); } #endif #include "a.h" using std::cout; using std::endl; int main() { cout << compare(1,2) << endl; cout << compare("123","asd") << endl; return 0; }
現(xiàn)在看上去沒什么問題,并且只有這兩個文件編譯通過且正常運行,如果多了一個包含a.h的文件a.cpp:
#include "a.h"
這里只有一句,但包含了a.h文件表示將其模板函數(shù)的定義包含,且特例化的函數(shù)類似一個普通函數(shù),則a.cpp、main.cpp包含多個相同函數(shù)的定義,因此在鏈接時有重定義問題。
解決方法:
使用內(nèi)聯(lián)inline聲明特例化的模板,則某些函數(shù)的定義可以在多個文件包含(一些函數(shù)的實現(xiàn)可能不支持內(nèi)聯(lián)):
template <> inline int compare(const char *a,const char *b) { return strcmp(a,b); }
另外一個是使用一個文件包含該頭文件,讓所有鏈接的文件只有一個特例化定義
第三個是定義普通同名函數(shù),通過重載調(diào)用非模板函數(shù)(在參數(shù)匹配級別相同時,非模板重載函數(shù)優(yōu)先調(diào)用)
// a.h #ifndef A_H #define A_H #include <iostream> #include <cstring> using std::less; template <typename T> int compare(const T v1,const T v2) { if (less<T>()(v1,v2)) { return -1; } else if (less<T>()(v2,v1)) { return 1; } else { return 0; } } int compare(const char *a,const char *b) ; // a.cpp #include "a.h" int compare(const char *a, const char *b) { std::cout << "const char*"<<std::endl; return strcmp(a,b); } // main.cpp 相同
如果有其他的方法請大家一起交流
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。