亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解C++ 模板編程

 更新時(shí)間:2020年09月08日 15:10:01   作者:tlanyan  
模板(template)是C++實(shí)現(xiàn)泛型(Generics)和元編程(Meta Programming)的基礎(chǔ)。本文拋磚引玉,簡(jiǎn)要介紹C++模板編程,不足之處敬請(qǐng)指正。

類型模板

類型模板包括函數(shù)模板和類模板,基本上是C++開發(fā)人員接觸模板編程的起點(diǎn)。

下面代碼演示了函數(shù)模板和類模板的使用方法:

// 函數(shù)模板
template<typename T>
T add(const T& a, const T& b) {
 return a + b;
}

// 類模板
template<typename T>
class Point {
private:
 T x[3];
 ...
};

類型模板以template開始聲明,尖括號(hào)內(nèi)的typename關(guān)鍵字可用class替代。類型模板中typenameclass具有相同含義,均表示參數(shù)類型。實(shí)踐中typename語義更廣泛,表示其后續(xù)的參數(shù)T是一個(gè)類型,不限定于類,建議使用。類型參數(shù)T可換成其他任意有意義的合法變量。

C++14新增變量模板:

// 變量模板
template<tyepename T>
constexpr T pi = T(3.1415926535897932385L);

尖括號(hào)之于模板猶如小括號(hào)之于函數(shù):函數(shù)通過小括號(hào)()定義和調(diào)用,模板使用尖括號(hào)<>定義(需template關(guān)鍵字聲明)和實(shí)例化。上面演示了類型模板定義,下面代碼介紹模板實(shí)例化:

int a = 1, b = 2;
// 實(shí)例化函數(shù)模板
std::cout << "add result:" << add<int>(a, b) << std::endl;

// 實(shí)例化類模板
auto p = Point<int>();

double radius = .5;
// 實(shí)例化變量模板
auto area = pi<double> * radius * radius;

同函數(shù)一樣,模板可以有默認(rèn)值:

// 默認(rèn)類型為int
template<typename T=int>
T add(const T& a, const T& b) {
 return a + b;
}

// 默認(rèn)類型為double
template<typename T=double>
class Point {
private:
 T x[3];
 ...
};

與函數(shù)不同,對(duì)于函數(shù)模板,如果能從參數(shù)推斷出模板類型,則可略去尖括號(hào)模板實(shí)例化參數(shù):

int a = 1, b = 2;
// 合法調(diào)用,編譯器能根據(jù)a b推斷出參數(shù)類型
std::cout << "add result:" << add(a, b) << std::endl;
// 等同于
std::cout << "add result:" << add<int>(a, b) << std::endl;

然而對(duì)于類模板,即使有默認(rèn)參數(shù),也不能省略尖括號(hào)(但是可以省去參數(shù)):

template<typename T=double>
struct Point {
 T x[3];
};

// 合法聲明
auto p = Point<double>();
// 合法聲明,類型使用默認(rèn)的double
auto p2 = Point<>();
// 非法聲明,缺少模板調(diào)用標(biāo)志尖括號(hào)
auto p3 = Point();

類型參數(shù)模板在實(shí)際中使用最多,STL庫(kù)中vector、map等容器、algorithm中的許多算法都用到了模板。

非類型參數(shù)模板

另一類常用模板是非參數(shù)模板,用來替代某個(gè)具體的值。例如:

// N維空間向量
template<int N>
struct Vector {
 double x[N];
};

// 實(shí)例化
auto v = Vector<100>();
...其他操作

需要注意的是,非類型參數(shù)模板能使用的類型十分有限,只有(signed/unsigned)整數(shù)、char和枚舉這幾種類型可用(參考switch語法)。

同類型模板一樣,非類型參數(shù)模板也可以有默認(rèn)值,但應(yīng)用到類模板實(shí)例化也不能省略尖括號(hào)。

類型模板和非類型參數(shù)模板可以結(jié)合一起用:

template<typename T, int N>
struct Point {
 T x[N];
};

類型模板解決了類型問題,非類型參數(shù)模板解決了值的問題,實(shí)際中應(yīng)用也十分廣泛。作為遞歸的經(jīng)典場(chǎng)景,斐波那契數(shù)列可以用非類型模板解決:

template<int N>
struct Fib {
 static constexpr int value = Fib<N-1>::value + Fib<N-2>::value;
};
// 模板特化
template<>
struct Fib<1> {
 static constexpr int value = 1;
};
// 模板特化
template<>
struct Fib<0> {
 static constexpr int value = 0;
};

// 調(diào)用
std::cout << "Fib(10): " << Fib<10>::value << std::endl;

這個(gè)例子出現(xiàn)了”模板特化”,接下來介紹。

模板特化/偏特化

定義模板后,希望在特定條件下使用單獨(dú)的模板,這便是模板特化。上文中斐波那契數(shù)列定義的template<int N> struct Fib是母模板,接下來又定義了0和1兩個(gè)特化模板(子模板),指示編譯器遇到Fib<0>和Fib<1>的情況,使用這兩組單獨(dú)定義。需要注意的是特化模板的template參數(shù)為空,具體模板參數(shù)放到了模板名稱處,類似于模板實(shí)例化。

對(duì)多個(gè)模板參數(shù)的情形,如果只特化某個(gè)模板參數(shù),便是偏特化。例如:

// 泛型模板定義
template<typename T1, typename T2> struct Add; 
// 特化模板
template<> struct Add<int, int> {...};
// 偏特化模板
template<typename T> struct Add<T, long> {....};

模板特化/偏特化類似于函數(shù)重載,能針對(duì)特殊情況進(jìn)行特別處理。

模板匹配與SFINAE

模板特化使得同一個(gè)模板名稱有了多個(gè)定義,代碼具體調(diào)用時(shí)會(huì)遇到模板匹配問題。理解模板匹配機(jī)制的關(guān)鍵便是SFINAE,這也是進(jìn)階模板編程的必備知識(shí)點(diǎn)。

SFINAE是Substitution failure is not an error的縮寫,翻譯過來便是:匹配(替換)失敗不是錯(cuò)誤。

怎么理解這句話呢?

對(duì)于上面的斐波那契數(shù)列數(shù)列代碼,編譯器遇到Fib<10>::value的代碼,(可能)先會(huì)嘗試匹配Fib<0>,發(fā)現(xiàn)匹配不上,這是一個(gè)Substitution failure,但不是error,所以編譯器繼續(xù)嘗試其他可能性。接著匹配Fib<1>,同樣發(fā)現(xiàn)匹配不上,忽略這個(gè)Substitution failure繼續(xù)嘗試Fib<N>,OK,這一次沒問題,編譯成功。

如果是Fib<-1>::value,編譯器達(dá)到最大遞歸深度也找不到一個(gè)合適的匹配模板,這是一個(gè)error,因此編譯失敗。

備注:理解上面的話需要對(duì)編譯過程稍加了解,編譯過程會(huì)輸出許多信息,編譯器一般只有遇到error才會(huì)終止編譯,比較常見的warning則不會(huì)。模板匹配中的Substitution可能連warning都算不上,不會(huì)影響編譯器繼續(xù)嘗試匹配

理解SFINAE是看懂稍微深?yuàn)W點(diǎn)模板代碼的基本功,重點(diǎn)便是:不怕你模板多,就怕找不到合適的模板。

兩階段編譯

有了模板(元)編程,C++源碼編譯可以分為前期和后期,構(gòu)成兩階段編譯。前期是模板的天下,編譯器掃描模板實(shí)例化語句,生成運(yùn)算結(jié)果和具體代碼;后期編譯器介入,再編譯生成機(jī)器碼。

模板代碼運(yùn)行在編譯期,因此有如下特點(diǎn):

  • 沒有實(shí)例化的模板代碼,即使有語法錯(cuò)誤,編譯器也不會(huì)檢查和報(bào)錯(cuò)。對(duì)按代碼行數(shù)考核KPI的C++碼農(nóng),這絕對(duì)是福音,新增template代碼十萬行,瞎編亂寫都可以,只要不實(shí)例化,永遠(yuǎn)能編譯通過,編譯后的文件大小(一般)不變,也不影響現(xiàn)有代碼運(yùn)行;
  • 對(duì)于常量,編譯前期直接計(jì)算,沒有運(yùn)行時(shí)開銷。上文中的斐波那契數(shù)列值在編譯期便已經(jīng)計(jì)算出來了;
  • 無法運(yùn)行期動(dòng)態(tài)調(diào)用代碼。例如下面的要求做不到:
template<int N>
struct Point {double x[N];};
// 根據(jù)輸入動(dòng)態(tài)生成類,無法實(shí)現(xiàn)和編譯成功
int n;
std::cin >> n;
auto p = new Point<n>();
  • 模板和多態(tài)/虛函數(shù)(理念)沖突。多態(tài)/虛函數(shù)的關(guān)鍵是運(yùn)行期動(dòng)態(tài)調(diào)用代碼,而模板在編譯期確定,因此兩者理念上是沖突的。所以,如果你想一個(gè)成員函數(shù)既是模板函數(shù),又是虛函數(shù),怎么做實(shí)現(xiàn)預(yù)期?

C/C++編譯有個(gè)預(yù)處理過程,只是做簡(jiǎn)單字符串替換,沒有具體運(yùn)算,與模板生成代碼不同

在編譯前期,除了模板代碼被解釋執(zhí)行,其他代碼信息都在,因此模板代碼擁有類似反射/自省的能力,這也是C++元編程功能強(qiáng)大的原因之一。

C++11中的變化

C++11帶來了許多新特性和重大更新,可以認(rèn)為C++11是一門新的語言。就模板來說,主要更新點(diǎn)如下:

1. 可以使用static constexpr int代替早期模板代碼中的enum。網(wǎng)上許多斐波那契數(shù)列代碼都是基于早期C++,一律使用enum方式定義字段;

2. 可以使用using代替typedef。這是using語句能力的重大更新,早期我們定義類型或者別名都需要typedef,自C++11開始,簡(jiǎn)單使用using就可以達(dá)到相同效果。

3. C++14引入了變量模板,上文已介紹。

模板優(yōu)缺點(diǎn)

上文根據(jù)自己理解和實(shí)踐簡(jiǎn)要介紹了C++模板編程的相關(guān)概念,本節(jié)總結(jié)一下C++模板的優(yōu)缺點(diǎn):

C++模板編程優(yōu)點(diǎn):

  • 減少代碼輸入,提高代碼重用和編程效率;
  • 支持鴨子類型(Duck typing)的特性,使用便利,功能強(qiáng)大;
  • 某些情況下能減少運(yùn)行期開銷;
  • 能實(shí)現(xiàn)元編程,C++高手必備之路;

C++模板編程缺點(diǎn):

  • 語法看起來是hack黑科技,代碼可讀性差,編寫繁瑣;
  • 模板代碼調(diào)試?yán)щy,生成的錯(cuò)誤信息也晦澀難懂。你可以還記得剛開始使用STL模板的map等數(shù)據(jù)類型報(bào)錯(cuò)的恐怖提示?
  • 編譯時(shí)間增加。

感謝閱讀,歡迎指正!

以上就是詳解C++ 模板編程的詳細(xì)內(nèi)容,更多關(guān)于C++ 模板編程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • C語言程序設(shè)計(jì)譚浩強(qiáng)第五版課后答案(第三章習(xí)題答案)

    C語言程序設(shè)計(jì)譚浩強(qiáng)第五版課后答案(第三章習(xí)題答案)

    這篇文章主要介紹了C語言程序設(shè)計(jì)譚浩強(qiáng)第五版課后答案(第三章習(xí)題答案),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2021-04-04
  • C++中std的使用及說明

    C++中std的使用及說明

    這篇文章主要介紹了C++中std的使用及說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • C語言模擬內(nèi)存函數(shù)分析之mencpy與memmove

    C語言模擬內(nèi)存函數(shù)分析之mencpy與memmove

    這篇文章主要介紹了C語言詳解如何模擬內(nèi)存函數(shù),用到了mencpy與memmove兩個(gè)函數(shù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2022-03-03
  • C++智能指針讀書筆記

    C++智能指針讀書筆記

    本篇隨筆僅作為個(gè)人學(xué)習(xí)《C++ Primer》智能指針一節(jié)后的部分小結(jié),抄書嚴(yán)重,伴隨個(gè)人理解。主要介紹shared_ptr、make_shared、weak_ptr的用法和聯(lián)系
    2015-11-11
  • VScode中C++頭文件問題的終極解決方法詳析

    VScode中C++頭文件問題的終極解決方法詳析

    最近使用VSCode編譯C/C++時(shí)發(fā)現(xiàn)了問題,下面這篇文章主要給大家介紹了關(guān)于VScode中C++頭文件問題的終極解決方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-08-08
  • C語言數(shù)獨(dú)游戲的求解方法

    C語言數(shù)獨(dú)游戲的求解方法

    這篇文章主要為大家詳細(xì)介紹了C語言數(shù)獨(dú)游戲的求解方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • 使用boost讀取XML文件詳細(xì)介紹

    使用boost讀取XML文件詳細(xì)介紹

    這篇文章主要介紹了使用boost讀取XML文件詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下
    2016-11-11
  • C++利用數(shù)組(一維/二維)處理批量數(shù)據(jù)的方法

    C++利用數(shù)組(一維/二維)處理批量數(shù)據(jù)的方法

    對(duì)于簡(jiǎn)單的問題,使用簡(jiǎn)單的數(shù)據(jù)類型就可以了,但是對(duì)于有些需要處理的數(shù)據(jù),只用以上簡(jiǎn)單的數(shù)據(jù)類型是不夠的,難以反映出數(shù)據(jù)的特點(diǎn),也難以有效的進(jìn)行處理,本文小編給大家介紹了C++利用數(shù)組(一維/二維)處理批量數(shù)據(jù)的方法,需要的朋友可以參考下
    2023-10-10
  • C++實(shí)現(xiàn)LeetCode(136.單獨(dú)的數(shù)字)

    C++實(shí)現(xiàn)LeetCode(136.單獨(dú)的數(shù)字)

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(136.單獨(dú)的數(shù)字),本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • c語言實(shí)現(xiàn)的hashtable分享

    c語言實(shí)現(xiàn)的hashtable分享

    哈希表效率高,眾所周知。應(yīng)用廣泛,php中大部分存儲(chǔ)使用的都是hashtable,包括變量,數(shù)組…如何使用c語言實(shí)現(xiàn)hashtable呢,現(xiàn)提供自己的思路,如有不妥之處,敬請(qǐng)賜教
    2014-01-01

最新評(píng)論