欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++模板全方位深入解讀

 更新時(shí)間:2022年06月06日 10:20:07   作者:林慢慢腦瓜子嗡嗡的  
人們需要編寫多個(gè)形式和功能都相似的函數(shù),因此有了函數(shù)模板來減少重復(fù)勞動(dòng);人們也需要編寫多個(gè)形式和功能都相似的類,于是?C++?引人了類模板的概念,編譯器從類模板可以自動(dòng)生成多個(gè)類,避免了程序員的重復(fù)勞動(dòng)

1.泛型編程

如何實(shí)現(xiàn)一個(gè)通用的交換函數(shù)?

這點(diǎn)函數(shù)重載可以做到,比如一下Swap函數(shù)的重載,分別重載了倆種不同參數(shù)類型的Swap

void Swap(int& x, int& y)
{
	int tmp = x;
	x = y;
	y = tmp;
}
void Swap(char& x, char& y)
{
	char tmp = x;
	x = y;
	y = tmp;
}

但是這也帶來了幾點(diǎn)不好的地方:

1.重載的函數(shù)僅僅是類型不同,代碼的復(fù)用率比較低,只要有新類型出現(xiàn),就需要增加對(duì)應(yīng)的函數(shù)

2.代碼的可維護(hù)性比較低,一個(gè)出錯(cuò)可能所有的重載均出錯(cuò)

那么有什么好的解決方法嗎?我們能否告訴編譯器一個(gè)模子,讓編譯器根據(jù)不同類型利用該模子自己去生成相應(yīng)的代碼呢?

當(dāng)然能,這就是泛型編程,即編寫與類型無關(guān)的通用代碼,這是代碼復(fù)用的一種手段。而模板是泛型編程的基礎(chǔ)。模板分為兩種,函數(shù)模板和類模板。

2.函數(shù)模板

概念

函數(shù)模板代表了一個(gè)函數(shù)家族,該函數(shù)模板與類型無關(guān),在使用時(shí)被參數(shù)化,根據(jù)實(shí)參類型產(chǎn)生函數(shù)的特定類型版本。

函數(shù)模板的格式

template<typename T1, typename T2,......,typename Tn>

即:返回值類型 函數(shù)名(參數(shù)列表){}

其中typename可以改成class(不能用struct)

例:

template <typename T>
void Swap(T& x, T& y)
{
	T tmp = x;
	x = y;
	y = tmp;
}

函數(shù)模板的原理

函數(shù)模板是一個(gè)藍(lán)圖,它本身并不是函數(shù),是編譯器用使用方式產(chǎn)生特定具體類型函數(shù)的模具。所以其實(shí)模板就是將本來應(yīng)該我們做的重復(fù)的事情交給了編譯器。

在編譯器編譯階段,對(duì)于模板函數(shù)的使用,編譯器需要根據(jù)傳入的實(shí)參類型來推演生成對(duì)應(yīng)類型的函數(shù)以供調(diào)用。比如:當(dāng)用double類型使用函數(shù)模板時(shí),編譯器通過對(duì)實(shí)參類型的推演,將T確定為double類型,然后產(chǎn)生一份專門處理double類型的代碼,對(duì)于字符類型也是如此。

函數(shù)模板的實(shí)例化

用不同類型的參數(shù)使用函數(shù)模板時(shí),稱為函數(shù)模板的實(shí)例化。模板參數(shù)的實(shí)例化分為兩種:隱式實(shí)例化和顯式實(shí)例化。

隱式實(shí)例化

讓編譯器自己推演函數(shù)參數(shù)的類型。

需要注意的是隱式實(shí)例化的參數(shù)一定要匹配,否則可能產(chǎn)生分歧導(dǎo)致編譯器無法識(shí)別。比如:

template<typename T>
T Add(const T& left, const T& right)
{
	return left + right;
}
int main()
{
	int a1 = 10, b1 = 20;
	double a2 = 10.0, b2 = 20.0;
	Add(a1, b2);
	return 0;
}

編譯器報(bào)錯(cuò),該語句不能通過編譯,因?yàn)闊o法確定T是int還是double。

如何處理?有兩種方式:

1.強(qiáng)制類型轉(zhuǎn)換!但值得注意的是,強(qiáng)轉(zhuǎn)會(huì)產(chǎn)生臨時(shí)變量,臨時(shí)變量是具有常性的,需要const修飾一下!

2.使用顯式實(shí)例化

顯式實(shí)例化

在函數(shù)名后的< >中指定模板參數(shù)的實(shí)際類型。

template<typename T>
T Add(const T& left, const T& right)
{
	return left + right;
}
int main()
{
	int a1 = 10, b1 = 20;
	double a2 = 10.0, b2 = 20.0;
	cout<<Add<int>(a1, b2)<<endl;
	return 0;
}

模板參數(shù)的匹配原則

1.一個(gè)非模板函數(shù)可以和一個(gè)同名的函數(shù)模板同時(shí)存在,而且該函數(shù)模板還可以被實(shí)例化為這個(gè)非模板函數(shù)。

2.對(duì)于非模板函數(shù)和同名函數(shù)模板,如果其他條件都相同,在調(diào)動(dòng)時(shí)會(huì)優(yōu)先調(diào)用非模板函數(shù)而不會(huì)從該模板產(chǎn)生出一個(gè)實(shí)例。如果模板可以產(chǎn)生一個(gè)具有更好匹配的函數(shù), 那么將選擇模板

3.模板函數(shù)不允許自動(dòng)類型轉(zhuǎn)換,但普通函數(shù)可以進(jìn)行自動(dòng)類型轉(zhuǎn)換

  • 如果有定義出來的函數(shù),且類型完全匹配調(diào)用時(shí)實(shí)參類型,則執(zhí)行定義出來的函數(shù).如果定義出來的函數(shù),不符合,則執(zhí)行模板推演.
  • 這部分沒啥難度,不再舉例說明,總的來說,對(duì)于函數(shù)調(diào)用的優(yōu)先級(jí)就是:完全匹配 >模板匹配 >轉(zhuǎn)換匹配。

3.類模板

(1) 類模板的定義格式

template<class T1, class T2, ..., class Tn>
class 類模板名
{
 // 類內(nèi)成員定義
}; 

注意:

類模板中函數(shù)放在類外進(jìn)行定義時(shí),需要加模板參數(shù)列表;

template <typename T>
class Stack
{
public:
	Stack(int capacity = 4)
		:_a(new T[capacity])
		,_top(0)
		,_capacity(capacity)
	{}
	~Stack()
	{
		delete[] _a;
		_top = _capacity = 0;
	}
	void Push(T x);
private:
	T* _a;
	int _top;
	int _capacity;
};
// 注意:類模板中函數(shù)放在類外進(jìn)行定義時(shí),需要加模板參數(shù)列表
template <typename T>
void Stack<T>::Push(T x)
{
	if (_top == _capacity)
	{
		_capacity *= 2;
		T* tmp = (T*)realloc(_a, sizeof(int) * _capacity);
		if (tmp == nullptr)
		{
			cout << "realloc fail" << endl;
			exit(-1);
		}
		_a = tmp;
	}
	_a[_top++] = x;
}

對(duì)于普通類,類名就是類型;對(duì)于類模板,類名不是類型,類型是Class < T >

(2) 類模板的實(shí)例化

類模板實(shí)例化與函數(shù)模板實(shí)例化不同,類模板實(shí)例化需要在類模板名字后跟 < >,然后將實(shí)例化的類型放在<>中即可,類模板名字不是真正的類,而實(shí)例化的結(jié)果才是真正的類。

int main()
{
	// Stack只是類名,不是類型,Stack<int>才是類型
	Stack<int> s1;
	Stack<char> s2;
	return 0;
}

4.非類型模板參數(shù)

類型參數(shù):就是在模板的參數(shù)列表中在class后面加上參數(shù)的類型名稱。

非類型參數(shù):就是用一個(gè)常量作為類(函數(shù))模板的一個(gè)參數(shù),在類(函數(shù))模板中可將該參數(shù)當(dāng)成常量來使用。

注意兩點(diǎn):

1.浮點(diǎn)數(shù)、類對(duì)象以及字符串是不允許作為非類型模板參數(shù)的。

2.非類型的模板參數(shù)必須在編譯期就能確認(rèn)結(jié)果。

template<class T =int, size_t N = 10>
class array
	{
		 private:
	 	 T _array[N];
		 size_t _size;
	 }

5.模板特化

(1)函數(shù)模板的特化

當(dāng)針對(duì)某一情景或者某一類型,函數(shù)模板無法滿足要求,模板需要有特殊的處理,這個(gè)時(shí)候就需要用到模板的特化。

比如咱們要比較兩個(gè)字符串是否相同:

template<class T>
bool IsEqual(T str1, T str2)
{
	return str1 == str2;
}
int main()
{
	char str1[] = "hello";
	char str2[] = "hello";
	if (IsEqual(str1, str2))
		cout << "true";
	else
		cout << "false";
}

上述代碼輸出false, 不滿足咱們的要求, 因?yàn)檎{(diào)用函數(shù)IsEqual()時(shí)傳遞過去的是兩個(gè)char*類型,他們兩個(gè)比較的不是字符串的內(nèi)容,而是指針的地址,所以返回false。

此時(shí)模板特化派上用場(chǎng)了:如果要比較char*, 可以用strcmp來對(duì)這個(gè)情況進(jìn)行特殊處理

template<>
bool IsEqual<char*>(char* str1, char* str2)
{
	return strcmp(str1, str2) == 0;
}

此時(shí)就返回true, 符合預(yù)期了。

函數(shù)模板的特化步驟:

  • 必須要先有一個(gè)基礎(chǔ)的函數(shù)模板
  • 關(guān)鍵字template后面接一對(duì)空的尖括號(hào)<>
  • 函數(shù)名后跟一對(duì)尖括號(hào),尖括號(hào)中指定需要特化的類型
  • 函數(shù)形參表: 必須要和模板函數(shù)的基礎(chǔ)參數(shù)類型完全相同

(2)類模板的特化

類也是同理,如果需要有特殊情景也需要特化處理

以如下類舉例(后邊全特化、偏特化都針對(duì)它):

template<class T1, class T2>
class test
{
public:
	test()
	{
		cout << "test<T1, T2>" << endl;
	}
private:
	T1 _x;
	T2 _y;
};

全特化

全特化即是將模板參數(shù)列表中所有的參數(shù)都確定化。

例如:

這里對(duì)test<int,double>版本特化

template<>
class test<int, double>
{
public:
	test()
	{
		cout << "test<int, double>" << endl;
	}
private:
	int _x;
	double _y;
};
int main()
{
	test<double, double> t1;
	test<int, double> t2;
}

偏特化

偏特化即是任何針對(duì)模版參數(shù)進(jìn)一步進(jìn)行條件限制設(shè)計(jì)的特化版本

偏特化有兩種表現(xiàn)方式,一種是部分參數(shù)特化,一種是參數(shù)修飾特化

部分參數(shù)特化

這里對(duì)第二個(gè)參數(shù)特化,只要第二個(gè)參數(shù)是double就會(huì)調(diào)用對(duì)應(yīng)特化版本

template<class T1>
class test<T1, double>
{
public:
	test()
	{
		cout << "test<T1, double>" << endl;
	}
private:
	T1 _x;
	double _y;
};
int main()
{
	test<double, double> t1;
	test<int, double> t2;
}

參數(shù)修飾特化

比如用指針或者引用來修飾類型,也可以進(jìn)行特化

template<class T1, class T2>
class test<T1*, T2*>
{
public:
	test()
	{
		cout << "test<T1*, T2*>" << endl;
	}
private:
	T1* _x;
	T2* _y;
};
int main()
{
	test<int*, double*> t;
}

6.模板的分離編譯

對(duì)于一個(gè)代碼量比較多的項(xiàng)目,通常都會(huì)采用聲明與定義分離的方法,比如在頭文件進(jìn)行聲明,在源文件完成代碼的實(shí)現(xiàn),最后通過鏈接的方法鏈接成單一的可執(zhí)行文件。但是C++的編譯器卻不支持模板的分離編譯,一旦進(jìn)行分離編譯,就會(huì)出現(xiàn)鏈接錯(cuò)誤。

問題分析

//頭文件a.h
template<class T>
bool IsEqual(const T& str1, const T& str2);
-------------
//源文件a.cpp
template<class T>
bool IsEqual(const T& str1, const T& str2)
{
	return str1 == str2;
}
--------------
//test.c
#include<iostream>
#include"a.h"
using namespace std;
int main()
{
	cout << IsEqual(3, 5);
	cout << IsEqual('a', 'b');
}

這里看上去是沒有問題的,但是涉及到了模板的實(shí)例化規(guī)則。

當(dāng)主函數(shù)調(diào)用這個(gè)函數(shù)的時(shí)候他就會(huì)去頭文件中找到函數(shù)的聲明,再通過聲明找到a.h中的實(shí)現(xiàn)。

但是對(duì)于模板卻并不會(huì)這樣,因?yàn)樯弦徽抡f過,模板的實(shí)例化只會(huì)在其第一次使用的時(shí)候才會(huì)進(jìn)行,例如這里IsEqual(3, 5),他就會(huì)去頭文件中尋找,但是頭文件中只有聲明,沒有定義,無法將其實(shí)例化。他又想通過找到a.cpp中的函數(shù)定義來進(jìn)行實(shí)例化,但是遺憾的是,a.cpp中只有IsEqual(const T& str1, const T& str2)的定義,沒有IsEqual(const int & str1, const int T& str2),因?yàn)樵赼.cpp中并沒有使用到該類型的實(shí)例,所以自然也不會(huì)為其實(shí)例化出來,這時(shí)test.cpp中就根本無法找到這個(gè)函數(shù)的實(shí)現(xiàn),就導(dǎo)致了鏈接失敗。

解決方法:

這個(gè)問題其實(shí)沒有什么完美的解決方法

  • 將聲明和定義放到同一個(gè)頭文件中。
  • 類模板顯式實(shí)例化。

到此這篇關(guān)于C++模板全方位深入解讀的文章就介紹到這了,更多相關(guān)C++模板內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • OpenCV圖像特征提取之Shi-Tomasi角點(diǎn)檢測(cè)算法詳解

    OpenCV圖像特征提取之Shi-Tomasi角點(diǎn)檢測(cè)算法詳解

    Harris角點(diǎn)檢測(cè)算法就是對(duì)角點(diǎn)響應(yīng)函數(shù)R進(jìn)行閾值處理,Shi-Tomasi原理幾乎和Harris一樣的,只不過最后計(jì)算角點(diǎn)響應(yīng)的公式發(fā)生了變化。本文將和大家詳細(xì)說說Shi-Tomasi角點(diǎn)檢測(cè)算法的原理與實(shí)現(xiàn),需要的可以參考一下
    2022-09-09
  • C++實(shí)現(xiàn)考勤管理系統(tǒng)

    C++實(shí)現(xiàn)考勤管理系統(tǒng)

    這篇文章主要介紹了C++實(shí)現(xiàn)考勤管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • FFmpeg實(shí)戰(zhàn)之分離出PCM數(shù)據(jù)

    FFmpeg實(shí)戰(zhàn)之分離出PCM數(shù)據(jù)

    PCM(Pulse?Code?Modulation,脈沖編碼調(diào)制)音頻數(shù)據(jù)是未經(jīng)壓縮的音頻采樣數(shù)據(jù)裸流,它是由模擬信號(hào)經(jīng)過采樣、量化、編碼轉(zhuǎn)換成的標(biāo)準(zhǔn)數(shù)字音頻數(shù)據(jù)。本文將通過FFmpeg實(shí)現(xiàn)分離PCM數(shù)據(jù),感興趣的可以了解一下
    2023-02-02
  • C語言數(shù)據(jù)結(jié)構(gòu)系列篇二叉樹的遍歷

    C語言數(shù)據(jù)結(jié)構(gòu)系列篇二叉樹的遍歷

    本章將會(huì)詳細(xì)講解二叉樹遍歷的四種方式,分別為前序遍歷、中序遍歷、后續(xù)遍歷和層序遍歷。在學(xué)習(xí)遍歷之前,會(huì)先帶大家回顧一下二叉樹的基本概念
    2022-02-02
  • 詳解PID控制器原理

    詳解PID控制器原理

    什么是 PID?它是一種在編程中使用的基本方法,如果正確調(diào)整,可以令人難以置信的有效和準(zhǔn)確,PID代表比例積分微分,3個(gè)單獨(dú)的部分連接在一起,雖然有時(shí)你不需要三個(gè)都使用。例如,您可以改為有P控制,PI控制或PD控制
    2021-06-06
  • c_str()的用法詳細(xì)解析

    c_str()的用法詳細(xì)解析

    c_str()就是把string類對(duì)象轉(zhuǎn)換成和c兼容的char *類型。這是為了與c語言兼容,在c語言中沒有string類型,故必須通過string類對(duì)象的成員函數(shù)c_str()把string 對(duì)象轉(zhuǎn)換成c中的字符串樣式
    2013-09-09
  • Windows下VScode實(shí)現(xiàn)簡單回聲服務(wù)的方法

    Windows下VScode實(shí)現(xiàn)簡單回聲服務(wù)的方法

    回聲服務(wù)端可以將客戶端傳來的信息,再原封不動(dòng)地發(fā)送給客戶端,因而得名 epoch 服務(wù)。接下來通過本文給大家介紹Windows下VScode實(shí)現(xiàn)簡單回聲服務(wù)的方法,感興趣的朋友一起看看吧
    2021-08-08
  • C語言詳解熱門考點(diǎn)結(jié)構(gòu)體內(nèi)存對(duì)齊

    C語言詳解熱門考點(diǎn)結(jié)構(gòu)體內(nèi)存對(duì)齊

    C?數(shù)組允許定義可存儲(chǔ)相同類型數(shù)據(jù)項(xiàng)的變量,結(jié)構(gòu)是?C?編程中另一種用戶自定義的可用的數(shù)據(jù)類型,它允許你存儲(chǔ)不同類型的數(shù)據(jù)項(xiàng),本篇讓我們來了解C?的結(jié)構(gòu)體內(nèi)存對(duì)齊
    2022-04-04
  • c語言snprintf函數(shù)的用法詳解

    c語言snprintf函數(shù)的用法詳解

    這篇文章主要給大家介紹了關(guān)于c語言snprintf函數(shù)用法的相關(guān)資料,snprintf()函數(shù)用于將格式化的數(shù)據(jù)寫入字符串,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-09-09
  • VSCode與Keil聯(lián)合開發(fā)STM32的流程

    VSCode與Keil聯(lián)合開發(fā)STM32的流程

    這篇文章主要介紹了VSCode與Keil聯(lián)合開發(fā)STM32的流程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02

最新評(píng)論