日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

C++11新特性之泛型编程与模板

發布時間:2025/3/8 c/c++ 30 豆豆
生活随笔 收集整理的這篇文章主要介紹了 C++11新特性之泛型编程与模板 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

  • 模板
    • 泛型編程
    • 函數模板
      • 普通函數模板
      • 成員函數模板
      • 函數模板重載
      • 模板函數的特化
    • 類模板
      • 類模板中的成員函數模板
      • 類模板的特化與偏特化
      • 類模板成員特化

模板

Template所代表的泛型編程是C++語言中的重要組成部分。

泛型編程

泛型編程(Generic Programming)是一種語言機制,通過它可以實現一個標準的容器庫。
像類一樣,泛型也是一種抽象數據類型,但是泛型不屬于面向對象,它是面向對象的補充和發展。
在面向對象編程中,當算法與數據類型有關時,面向對象在對算法的抽象描述方面存在一些缺陷。

首先我們先來了解什么是泛型編程,看下面的例子。

比如對棧的描述:

class stack {push(參數類型) //入棧算法pop(參數類型) //出棧算法}

如果把上面的偽代碼看作算法描述,沒問題,因為算法與參數類型無關。但是如果把它寫成可編譯的源代碼,
就必須指明是什么類型,否則是無法通過編譯的。使用重載來解決這個問題,即對N種不同的參數類型寫N個
push和pop算法,這樣是很麻煩的,代碼也無法通用。

若對上面的描述進行改造如下:
首先指定一種通用類型T,不具體指明是哪一種類型。

class stack<參數模板 T> {push(T) //入棧算法pop(T) //出棧算法}

這里的參數模板T相當于一個占位符,當我們實例化類stack時,T會被具體的數據類型替換掉。
若定義對象S為statc類型,在實例化S時若我們將T指定int型則:
這時候類S就成為:

class S {push(int) //入棧算法pop(int) //出棧算法 }

這時我可以稱class stack<參數模板 T>是類的類,通過它可以生成具體參數類型不同的類。

泛型在C++中的應用:
==泛型在C++中的主要實現為模板函數和模板類。==

函數模板

把處理不同類型的公共邏輯抽象成函數,就得到了函數模板。

函數模板的格式:

template <class 形參名,class 形參名,......> 返回類型 函數名(參數列表) {函數體 }

其中templateclass是關鍵字,當然class可以使用typename關鍵字代替,兩者之間一點區別都沒有,這個是真的。
<>括號中的參數叫模板形參,模板形參和函數形參很相像,模板形參不能為空。

普通函數模板

template<typename T> int compare(const T& left, const T& right) {if (left < right) {return -1; }if (right < left) {return 1; }return 0; }template<class T=double> void processValue( T& value ) {std::cout << value << std::endl; }int main() {int a=0;processValue(a);std::cout<< compare<int>(3,5) << std::endl;std::cout<< compare(3,5) << std::endl;return 0; }輸出結果為: 0 -1 -1

由上面的例子我們可以看出,除了直接為函數模板指定類型參數之外,我們還可以讓編譯器從傳遞給函數的實參推斷類型參數,這一功能被稱為模板實參推斷,形參T可以自動推到出類型,當我們傳入的是int類型時,T此時會被替換成int。函數模板支持默認的形參類型,如上面的template。

成員函數模板

不僅普通函數可以定義為模板,類的成員函數也可以定義為模板。

class Printer { public:template<typename T>void print(const T& t) {cout << t <<endl;} };int main() {Printer p;p.print<const char*>("abc");p.print(1);return 0; }輸出結果: abc 1

使用的方式和普通函數模板沒有什么兩樣。

總結:
1) 函數模板并不是真正的函數,它只是C++編譯生成具體函數的一個模子。
2) 函數模板本身并不生成函數,實際生成的函數是替換函數模板的那個函數。這種替換是編譯期就綁定的。
3) 函數模板不是只編譯一份滿足多重需要,而是為每一種替換它的函數編譯一份。
4) 函數模板不允許自動類型轉換。
5) 函數模板不可以設置默認模板實參。比如template 不可以。
6) 函數模板的模板形參不能為空。

補充:

為什么成員函數模板不能是虛函數(virtual)?

這是因為c++ compiler在parse一個類的時候就要確定vtable的大小,如果允許一個虛函數是模板函數,那么compiler就需要在parse這個類之前掃描所有的代碼,找出這個模板成員函數的調用(實例化),然后才能確定vtable的大小,而顯然這是不可行的,除非改變當前compiler的工作機制。

函數模板和模板函數是什么?
函數模板的重點是模板。表示的是一個模板,專門用來生產函數。
模板函數的重點是函數。表示的是由一個模板生成而來的函數。

當返回值類型也是參數時
當一個模板函數的返回值類型需要用另外一個模板參數表示時,你無法利用實參推斷獲取全部的類型參數,這時有兩種解決辦法:

  • 返回值類型與參數類型完全無關,那么就需要顯示的指定返回值類型,其他的類型交給實參推斷。
    注意:此行為與函數的默認實參相同,我們必須從左向右逐一指定。
template<typename T1, typename T2, typename T3> T1 sum(T2 v2, T3 v3) {return static_cast<T1>(v2 + v3); }auto ret = sum<long>(1L, 23); //指定T1, T2和T3交由編譯器來推斷template<typename T1, typename T2, typename T3> T3 sum_alternative(T1 v1, T2 v2) {return static_cast<T1>(v1 + v2); } auto ret = sum_alternative<long>(1L, 23); //error,只能從左向右逐一指定 auto ret = sum_alternative<long,int,long>(1L,23); //ok, 誰叫你把最后一個T3作為返回類型的呢?int main() {int a = 3;auto ret = sum(1.3443, 23); //指定T1, T2和T3交由編譯器來推斷,編譯錯誤,必須指定返回類型T1auto ret = sum<double>(1.3443, 23); //編譯通過std::cout<< ret << std::endl; //結果為24.3443auto ret1 = sum_alternative<double>(1.3443, 23); //error,只能從左向右逐一指定std::cout<< ret1 << std::endl;auto ret2 = sum_alternative<double,int,double>(1.3443,23); //ok, 誰叫你把最后一個T3作為返回類型的呢?std::cout<< ret2 << std::endl; //結果24.3443return 0; }
  • 返回值類型可以從參數類型中獲得,那么把函數寫成尾置返回類型的形式,就可以愉快的使用實參推斷了。
template<typename T> auto sum(T beg, T end) -> decltype(*beg) {decltype(*beg) ret = *beg;for (T it = beg+1; it != end; it++) {ret = ret + *it;}return ret; }int main() {std::vector<int> v = {1, 2, 3, 4};auto s = sum(v.begin(), v.end()); //s = 10std::cout << s << std::endl; //結果為10return 0; }

函數模板重載

函數模板之間,函數模板與普通函數之間可以重載。編譯器會根據調用時提供的函數參數,調用能夠處理這一類型的最特殊的版本。在特殊性上,一般按照如下順序考慮:
1. 普通函數
2. 特殊模板(限定了T的形式的,指針、引用、容器等)
3. 普通模板(對T沒有任何限制的)

template<typename T> void func(T& t) { //通用模板函數cout << "In generic version template " << t << endl; }template<typename T> void func(T* t) { //指針版本cout << "In pointer version template "<< *t << endl; }void func(string* s) { //普通函數cout << "In normal function " << *s << endl; }int i = 10; func(i); //調用通用版本,其他函數或者無法實例化或者不匹配 func(&i); //調用指針版本,通用版本雖然也可以用,但是編譯器選擇最特殊的版本 string s = "abc"; func(&s); //調用普通函數,通用版本和特殊版本雖然也都可以用,但是編譯器選擇最特化的版本 func<>(&s); //調用指針版本,通過<>告訴編譯器我們需要用template而不是普通函數

模板函數的特化

有時通用的函數模板不能解決個別類型的問題,我們必須對此進行定制,這就是函數模板的特化。函數模板的特化必須把所有的模版參數全部指定。

template<> void func(int i) {cout << "In special version for int "<< i << endl; }int main() {int i = 10;func(i); //調用特化版本return 0; }

類模板

類模板也是公共邏輯的抽象,通常用來作為容器(例如:vector)或者行為的封裝。

類模板的格式:
template

#include <iostream> #include <vector> #include <string> #include <sstream>template<typename T> class Printer { public:explicit Printer(const T& param):t(param){}//右值引用string&& to_string(){std::stringstream ss;ss << t;return std::move(string(ss.str()));} void print() {cout << t << endl;} private:T t; };int main() {Printer p(1); //errorPrinter<int> p(3); //okstd::string str = p.to_string();std::cout << str << std::endl; //結果為3return 0; }

與函數模板不同,類模板不能推斷實例化。所以你只能顯示指定類型參數使用Printer p(3),而不能讓編譯器自行推斷。

類模板的成員函數既可以定義在內部,也可以定義在外部。定義在內部的被隱式聲明為inline,定義在外部的類名之前必須加上template的相關聲明。

類模板中的成員函數模板

我們還可以把類模板和函數模板結合起來,定義一個含有成員函數模板的類模板。

template<typename T> class Printer { public:explicit Printer(const T& param):t(param){}//成員函數模板template<typename U>void add_and_print(const U& u); private:T t; };//注意這里要有兩層template的說明 template<typename T> template<typename U> void Printer<T>::add_and_print(const U& u) {cout << t + u << endl; }Printer<int> p(42); p.add_and_print(1.1); //自動推斷U為double,打印出43.1

類模板成員函數實例化

為了節省資源,類模板實例化時并不是每個成員函數都實例化了,而是使用到了哪個成員函數,那個成員函數才實例化。

template<typename T> class Printer { public:explicit Printer(const T& param):t(param){}void print() {cout << t << endl;}private:T t;};class empty{};empty e; Printer<empty> p(e); //ok

雖然成員函數print無法通過編譯,但是因為沒有使用到,也就沒有實例化print,所以沒有觸發編譯錯誤。

類模板的特化與偏特化

就像函數模板重載那樣,你可以通過特化(偏特化)類模板來為特定的類型指定你想要的行為。類模板的特化(偏特化)只需要模板名稱相同并且特化列表<>中的參數個數與原始模板對應上即可,模板參數列表不必與原始模板相同模板名稱相同。一個類模板可以有多個特化,與函數模板相同,編譯器會自動實例化那個最特殊的版本。

#include <typeinfo>template<typename T> //基本模板 class S { public:void info() { printf("In base template\n"); } };template<> //特化 class S<int> { public:void info() {printf("In int specialization\n");} };template<typename T> //偏特化 class S<T*> { public:void info() {printf("In pointer specialization\n");} };template<typename T, typename U> //另外一個偏特化 class S<T(U)> { public:void info() {std::cout << typeid(T).name() << std::endl; std::cout << typeid(U).name() << std::endl;printf("In function specialization\n");} };int func(int i) {return 2 * i; }S<float> s1; s1.info(); //調用base模板 S<int> s2; s2.info(); //調用int特化版本 S<float*> s3; s3.info(); //調用T*特化版本 S<decltype(func)> s4; s4.info(); //調用函數特化版本

提供了所有類型實參的特化是完全特化,只提供了部分類型實參或者T的類型受限(例如:T)的特化被認為是不完整的,所以也被稱為偏特化。完全特化的結果是一個實際的class,而偏特化的結果是另外一個同名的模板。*

類模板成員特化

除了可以特化類模板之外,還可以對類模板中的成員函數和普通靜態成員變量進行特化。

template<typename T> class S { public:void info() {printf("In base template\n");}static int code; };template<typename T> int S<T>::code = 10;template<> int S<int>::code = 100; //普通靜態成員變量的int特化template<> void S<int>::info() { //成員函數的int特化printf("In int specialization\n"); } S<float> s1; s1.info(); //普通版本 printf("Code is: %d\n", s1.code); //code = 10S<int> s2; s2.info(); //int特化版本 printf("Code is: %d\n", s2.code); //code = 100

*補充:*

類模板的重點是模板。表示的是一個模板,專門用于產生類的模

例如:

template <typename T> class Vector { … };

模板類的重點是類。表示的是由一個模板生成而來的類
例如:

Vector <int> 、Vector <char> 、Vector < Vector <int> > 、Vector <Shape*> ……//全是模板類

總結

以上是生活随笔為你收集整理的C++11新特性之泛型编程与模板的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。