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

C++印刷模板使用方法詳解

 更新時間:2022年11月10日 11:39:32   作者:The s.k.y.  
模板是C++支持參數(shù)化多態(tài)的工具,使用模板可以使用戶為類或者函數(shù)聲明一種一般模式,使得類中的某些數(shù)據(jù)成員或者成員函數(shù)的參數(shù)、返回值取得任意類型

在了解string之前,我們需要了解模板等等的一些鋪墊知識,讓我們開始吧!

一、泛型編程

泛型編程是什么意思呢?我們通過下面的例子來具體了解:

void Swap(int& left, int& right)
{
	int temp = left;
	left = right;
	right = temp;
}
void Swap(double& left, double& right)
{
	double temp = left;
	left = right;
	right = temp;
}
int main()
{
	int a = 1, b = 2;
	Swap(a, b);	
    double c=1.33,d=2.33;
    Swap(a,b)
}

就拿交換函數(shù)來說,當(dāng)我們交換不同類型的變量的值,那就需要不停的寫交換函數(shù)的重載,這樣代碼復(fù)用率就較低,那我們能不能創(chuàng)造一個模板呢??

一個Swap的模板,但是我可以用不同的類型去實(shí)現(xiàn)這個模板,繼而試用它。

如果在 C++ 中,也能夠存在這樣一個 模具 ,通過給這個模具中 填充不同材料 ( 類型 ) ,來 獲得不同材料的鑄件 ( 即生成具體類型的代碼)。 泛型編程:編寫與類型無關(guān)的通用代碼,是代碼復(fù)用的一種手段。模板是泛型編程的基礎(chǔ)。

二、模板(初階)

模板分為:函數(shù)模板和類模板

1.函數(shù)模板

1.單參數(shù)類型

函數(shù)模板代表了一個函數(shù)家族,該函數(shù)模板與類型無關(guān),在使用時被參數(shù)化,根據(jù)實(shí)參類型產(chǎn)生函數(shù)的特定 類型版本。 就拿Swap來說: typename 是 用來定義模板參數(shù) 關(guān)鍵字,T是類型(也可以用class,(class T))

template<typename T>
void Swap(T& left, T& right)
{
	T temp = left;
	left = right;
	right = temp;
}
int main()
{
	    int a = 1, b = 2;
	    Swap(a, b);
        int x = 1, y = 2;
		Swap(x, y);
		double m = 1.1, n = 2.2;
		Swap(m, n);
		char p = 'a', q = 'b';
		Swap(p, q);
        Swap(m,a);//不同類型
}

那么,具體是怎樣實(shí)現(xiàn)的呢?

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

編譯器通過類型推演,將函數(shù)模板進(jìn)行實(shí)例化,對應(yīng)的T就會替換成具體的類型,模板實(shí)例化是用幾個實(shí)例化幾個,不是所有不同類型都提前模板實(shí)例化。

1.當(dāng)變量類型相同,但是變量不同,調(diào)用Swap();模板實(shí)例化只會實(shí)例化一個,因?yàn)殡m然變量不同,但類型相同,模板實(shí)例化就是將T換成具體的類型。

2.當(dāng)Swap(m,a),變量是不同類型時,會發(fā)生什么??

因?yàn)樵谕蒲輛oid Swap(T& left, T& right);時,T的類型不明確,就會發(fā)生錯誤(推演報錯),直接報錯

但如果不用模板,我們自己這樣:

void Swap(int& left, int& right)
{
	int temp = left;
	left = right;
	right = temp;
}
int main()
{
  int a=2;
  double b=2.22;
  Swap(a,b);
}

可能有人就會想:在Swap(a,b)中,會不會a和b發(fā)生飲食類型轉(zhuǎn)化呢?較小的類型轉(zhuǎn)化成較大的類型。

當(dāng)然不會:隱式類型轉(zhuǎn)化只有在 賦值:b=3;(產(chǎn)生臨時變量);函數(shù)傳參的時候(產(chǎn)生臨時變量),才會發(fā)生隱式類型轉(zhuǎn)化。

函數(shù)形參是引用,當(dāng)類型是引用時,我們就要小心:是否會發(fā)生權(quán)限放大?當(dāng)b傳值時,中間的臨時變量具有常性(只讀),而形參是可讀可寫,權(quán)限就會放大,也是不可以通過的,除非加了const,但是加了const就無法交換了,所以這樣還是行不通的!

自動推演實(shí)例化和顯式實(shí)例化:

template<class T>
T Add(const T& left, const T& right)
{
	return left + right;
}
int main()
{
	int a1 = 10, a2 = 20;
	double d1 = 10.1, d2 = 20.2;
	// 自動推演實(shí)例化
	cout << Add(a1, a2) << endl;
	cout << Add(d1, d2) << endl;
	cout << Add((double)a1, d2) << endl; //強(qiáng)制類型轉(zhuǎn)化也是產(chǎn)生臨時變量,不是改變a1
	cout << Add(a1, (int)d2) << endl;
	// 顯示實(shí)例化
	cout << Add<double>(a1, d2) << endl;//隱式類型轉(zhuǎn)化
	cout << Add<int>(a1, d2) << endl;
	return 0;
}

在自動推演實(shí)例化中,必須強(qiáng)轉(zhuǎn),不然還是和之前問題一樣,該語句不能通過編譯,因?yàn)樵诰幾g期間,當(dāng)編譯器看到該實(shí)例化時,需要推演其實(shí)參類型通過實(shí)參a1將T推演為int,通過實(shí)參d1將T推演為double類型,但模板參數(shù)列表中只有一個T,編譯器無法確定此處到底該將T確定為int 或者 double類型而報錯。 (推演報錯)

不強(qiáng)轉(zhuǎn)情況:顯示實(shí)例化,:在函數(shù)名后的<>中指定模板參數(shù)的實(shí)際類型(我讓你怎么來你就怎么來?。?/p>

在函數(shù)名后加入了指定模板參數(shù)后,就會在實(shí)例化時,T直接是指定的類型,這樣就會發(fā)生隱式類型轉(zhuǎn)換。

注意:在模板中,編譯器一般不會進(jìn)行類型轉(zhuǎn)換操作,因?yàn)橐坏┺D(zhuǎn)化出問題,編譯器就需要背黑鍋 Add(a1, d1); 此時有兩種處理方式:1. 用戶自己來強(qiáng)制轉(zhuǎn)化 2. 使用顯式實(shí)例化

2.多參數(shù)類型

template<typename T1,typename T2),T1,T2為不同類型,當(dāng)然參數(shù)個數(shù)大于等于2

template<class t1,class t2>
t1 Add(const t1& left, const t2& right)
{
	return left + right;
}
int main()
{
	int a = 1, b = 2;
	double m = 2.22, n = 3.33;
	cout << Add(a, b) << endl;
	cout << Add(m, n) << endl;
	cout << Add(a, m) << endl;
	cout << Add(n, b) << endl;
}

此時,當(dāng)Add(不同類型時),就不會發(fā)生推演錯誤,你是什么類型就會推演成什么模板函數(shù)。

3.模板函數(shù)和自定義函數(shù)

當(dāng)模板函數(shù)和自己實(shí)現(xiàn)的函數(shù)是否可以同時存在時?

//專門處理int的加法函數(shù)
int Add(int left, int right) 
{
	return left + right;
}
// 通用加法函數(shù)
template<class T>
T Add(T left, T right) 
{
	return left + right;
}
int main()
{
	int a = 1, b = 2;
	Add(a, b);
	Add<int>(a, b);
	return 0;
}

當(dāng)自己寫的函數(shù)和模板函數(shù)同時存在時,二者不會沖突,在之前我們講過他們的函數(shù)名修飾規(guī)則是不同的。

同時存在,且調(diào)用時,首先會調(diào)用自己寫的函數(shù)。因?yàn)槟0搴瘮?shù)相當(dāng)于一個半成品,他需要推演實(shí)例化才會生成具體的函數(shù),所以當(dāng)然先使用自己實(shí)現(xiàn)的。

如果一定要使用模板函數(shù)的話,就需要顯示實(shí)例化:Add<int>(a,b);

這就叫泛型編程,與具體的類型無關(guān)!

2.類模板

類模板與函數(shù)模板不同的是:類模板統(tǒng)一顯式實(shí)例化,不需要推演,或者說沒有推演的時機(jī),而函數(shù)模板實(shí)參傳遞形參時,就會發(fā)生推演實(shí)例化。

格式:

template<typename T>
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = (T*)malloc(sizeof(T)*capacity);
		if (_a == nullptr)
		{
			perror("malloc fail");
			exit(-1);
		}
		_top = 0;
		_capacity = capacity;
	}
	 ......
private:
	T* _a;
	int _top;
	int _capacity;
};
int main()
{
	// 顯示實(shí)例化
	Stack<double> st1; // double
	st1.Push(1.1);
	Stack<int> st2; // int
	st2.Push(1);
	// s1,s2是同一個類模板實(shí)例化出來的,但是模板參數(shù)不同,他們就是不同類型
	return 0;
}

可能有人會問:s1=s2; 會不會發(fā)生隱式類型轉(zhuǎn)換呢?當(dāng)然不會,隱式類型轉(zhuǎn)換只有在類型相近才會發(fā)生。

接下來創(chuàng)建一個數(shù)組類模板:

namespace mj
{
	template<class T>
	class array
	{
	public:
		inline T& operator[](size_t i)  //這里引用做返回值的目的是除了減少拷貝構(gòu)造,還有可以修改返回值,直接可以修改數(shù)組里的值
		{
			assert(i < N);  //嚴(yán)格控制越界訪問的情況
			return _a[i];
		}
	private:
		T _a[N];
	};
}
int main()
{
	mj::array<int> a1;
	for (size_t i = 0; i < N; ++i)
	{
		//相當(dāng)于: a1.operator[](i)= i;
		a1[i] = i;
	}
	for (size_t i = 0; i < N; ++i)
	{
		// a1.operator[](i)
		cout << a1[i] << " ";
	}
	cout << endl;
	for (size_t i = 0; i < N; ++i)
	{
		a1[i]++;
	}
	for (size_t i = 0; i < N; ++i)
	{
		cout << a1[i] << " ";
	}
	cout << endl;
	return 0;
}

我們可以發(fā)現(xiàn),類對象居然也可以使用數(shù)組那一套了??當(dāng)然是取決于運(yùn)算符重載。

他與普通數(shù)組最大的區(qū)別是:

1. 普通數(shù)組對于數(shù)組越界的這種情況,只能隨機(jī)的抽查!而我們自己實(shí)現(xiàn)的類模板可以嚴(yán)格的控制越界訪問這種情況!別說越界修改,越界訪問都不行!

2.效率上因?yàn)閇]是運(yùn)算符重載,使用就會調(diào)用函數(shù)開辟棧幀,但是若定義到類中,并且加inline,就對于效率來說,那真是完美!

3.模板不支持分離編譯

我們在實(shí)現(xiàn)數(shù)據(jù)結(jié)構(gòu)的時候,是不是會經(jīng)常去分幾個文件去實(shí)現(xiàn)不同模塊的功能?

(個人習(xí)慣).h文件中,我們寫聲明;.cpp文件中,我們寫定義;test.cpp中,我們測試使用

但今天學(xué)習(xí)的模板就不能這么做了?。【唧w不能怎么做,我們上代碼:

如果這樣寫的話,他就會報鏈接錯誤(就是在使用時找不到定義)

我們知道,在預(yù)處理階段,就會將.h頭文件展開,test.cpp中只有聲明,在調(diào)用函數(shù)時,就會去找他的地址(call stack()),那么在編譯的時候,編譯器允許只有聲明沒有函數(shù),相當(dāng)于你可以先給他一個承諾,兌不兌現(xiàn)后面再說。

但在鏈接的時候,test.cpp中,卻不能找到它的地址,這是為什么??這就是模板和其他的區(qū)別!

鏈接錯誤原因:

.cpp中的定義,不是實(shí)例化模板,他只是一個模板,沒有任何實(shí)例化成任何類型。所以你在使用類模板的時候,壓根就找不到它的定義,當(dāng)然也找不到地址了,這不就鏈接錯誤了嗎?

看上圖:stack<int> st; 顯示實(shí)例化,但是.h中只有聲明,test.cpp用的地方實(shí)例化了,但是定義的地方stack.cpp卻沒有實(shí)例化,只是一個模板。

用的地方在實(shí)例化,但是有聲明,沒有定義;

定義的地方?jīng)]有實(shí)例化。

解決方法:

那轉(zhuǎn)來轉(zhuǎn)去就是一個問題:stack.cpp中定義沒有實(shí)例化?。?/p>

辦法一:

你沒有實(shí)例化,我給你補(bǔ)上:在定義后面加一個實(shí)例化

template<class T>
Stack<T>::Stack(int capacity = 4)
{
	cout << "Stack(int capacity = )" << capacity << endl;
	_a = (T*)malloc(sizeof(T)*capacity);
	if (_a == nullptr)
	{
		perror("malloc fail");
		exit(-1);
	}
	_top = 0;
	_capacity = capacity;
	template
	class Stack<int>;

但是就會有另一個問題,我如果使用的時候,創(chuàng)建不同類型,那模板實(shí)例化就要有不同類型,那就要一直補(bǔ)實(shí)例化,總不肯用一個補(bǔ)一個吧。

方法二:

那就是模板的編譯不分離:(不要將定義和聲明一個到.cpp,一個到.h)

當(dāng)放在一個文件中時,在編譯時,.h 文件展開后,定義和聲明都在test.cpp中,那直接就會完成模板實(shí)例化,就有了函數(shù)地址,不需要再去鏈接了。

鏈接:只有聲明沒有定義才會到處去找定義。

那有人就會問,加inline可以嗎?

inline當(dāng)然不可以,加了inline后,直接不產(chǎn)生符號表,還存在什么地址嗎?

直接放類中也不行,當(dāng)數(shù)據(jù)量大的時候,都擠到一推,代碼閱讀性很差,會傻傻搞不清!

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

相關(guān)文章

  • C++如何實(shí)現(xiàn)DNS域名解析

    C++如何實(shí)現(xiàn)DNS域名解析

    這片文章介紹了C++如何實(shí)現(xiàn)DNS域名解析,還有對相關(guān)技術(shù)的介紹,代碼很詳細(xì),需要的朋友可以參考下
    2015-07-07
  • C語言實(shí)例梳理講解常用關(guān)鍵字的用法

    C語言實(shí)例梳理講解常用關(guān)鍵字的用法

    關(guān)鍵字是C語言非常重要的一部分,熟練的掌握和使用關(guān)鍵字有助于我們更加熟悉了解C語言,同時C語言的關(guān)鍵字也是面試筆試中??嫉膬?nèi)容。C語言的關(guān)鍵字共有32個,但并不是每個關(guān)鍵字都有坑,本篇文章將通過理論聯(lián)系實(shí)際的方式為大家講解C語言中易混易錯以及??嫉囊恍╆P(guān)鍵字
    2022-05-05
  • 對for循環(huán)中表達(dá)式和循環(huán)體的執(zhí)行順序詳解

    對for循環(huán)中表達(dá)式和循環(huán)體的執(zhí)行順序詳解

    今天小編就為大家分享一篇對for循環(huán)中表達(dá)式和循環(huán)體的執(zhí)行順序詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-06-06
  • MFC實(shí)現(xiàn)字幕滾動效果

    MFC實(shí)現(xiàn)字幕滾動效果

    這篇文章主要為大家詳細(xì)介紹了MFC實(shí)現(xiàn)滾動字幕,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • OpenCV實(shí)現(xiàn)拼接圖像的簡單方法

    OpenCV實(shí)現(xiàn)拼接圖像的簡單方法

    這篇文章主要為大家詳細(xì)介紹了OpenCV實(shí)現(xiàn)拼接圖像的簡單方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • C語言實(shí)現(xiàn)個人通訊錄管理系統(tǒng)

    C語言實(shí)現(xiàn)個人通訊錄管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)個人通訊錄管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-12-12
  • C語言中#define定義的標(biāo)識符和宏實(shí)例代碼

    C語言中#define定義的標(biāo)識符和宏實(shí)例代碼

    C語言中,可以用#define定義一個標(biāo)識符來表示一個常量,下面這篇文章主要給大家介紹了關(guān)于C語言中#define定義的標(biāo)識符和宏的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-03-03
  • C++?Boost?Assign超詳細(xì)講解

    C++?Boost?Assign超詳細(xì)講解

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-12-12
  • C語言怎么連接兩個數(shù)組的內(nèi)容你知道嗎

    C語言怎么連接兩個數(shù)組的內(nèi)容你知道嗎

    這篇文章主要為大家介紹了C語言怎么連接兩個數(shù)組的內(nèi)容,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • C++ 實(shí)現(xiàn)PE文件特征碼識別的步驟

    C++ 實(shí)現(xiàn)PE文件特征碼識別的步驟

    PE文件就是我們常說的EXE可執(zhí)行文件,針對文件特征的識別可以清晰的知道該程序是使用何種編程語言實(shí)現(xiàn)的,前提是要有特征庫,PE特征識別有多種形式,第一種是靜態(tài)識別,第二種則是動態(tài)識別,我們經(jīng)常使用的PEID查殼工具是基于靜態(tài)檢測的方法。
    2021-06-06

最新評論