簡單掌握C++中的函數(shù)模板
1.函數(shù)模板的聲明和模板函數(shù)的生成
1.1函數(shù)模板的聲明
函數(shù)模板可以用來創(chuàng)建一個通用的函數(shù),以支持多種不同的形參,避免重載函數(shù)的函數(shù)體重復(fù)設(shè)計(jì)。它的最大特點(diǎn)是把函數(shù)使用的數(shù)據(jù)類型作為參數(shù)。
函數(shù)模板的聲明形式為:
template<typename 數(shù)據(jù)類型參數(shù)標(biāo)識符>
<返回類型><函數(shù)名>(參數(shù)表)
{
函數(shù)體
}
其中,template是定義模板函數(shù)的關(guān)鍵字;template后面的尖括號不能省略;typename(或class)是聲明數(shù)據(jù)類型參數(shù)標(biāo)識符的關(guān)鍵字,用以說明它后面的標(biāo)識符是數(shù)據(jù)類型標(biāo)識符。這樣,在以后定義的這個函數(shù)中,凡希望根據(jù)實(shí)參數(shù)據(jù)類型來確定數(shù)據(jù)類型的變量,都可以用數(shù)據(jù)類型參數(shù)標(biāo)識符來說明,從而使這個變量可以適應(yīng)不同的數(shù)據(jù)類型。例如:
template<typename T>
T fuc(T x, int y)
{
T x;
//……
}
如果主調(diào)函數(shù)中有以下語句:
double d; int a; fuc(d,a);
則系統(tǒng)將用實(shí)參d的數(shù)據(jù)類型double去代替函數(shù)模板中的T生成函數(shù):
double fuc(double x,int y)
{
double x;
//……
}
函數(shù)模板只是聲明了一個函數(shù)的描述即模板,不是一個可以直接執(zhí)行的函數(shù),只有根據(jù)實(shí)際情況用實(shí)參的數(shù)據(jù)類型代替類型參數(shù)標(biāo)識符之后,才能產(chǎn)生真正的函數(shù)。
關(guān)鍵字typename也可以使用關(guān)鍵字class,這時數(shù)據(jù)類型參數(shù)標(biāo)識符就可以使用所有的C++數(shù)據(jù)類型。
1.2.模板函數(shù)的生成
函數(shù)模板的數(shù)據(jù)類型參數(shù)標(biāo)識符實(shí)際上是一個類型形參,在使用函數(shù)模板時,要將這個形參實(shí)例化為確定的數(shù)據(jù)類型。將類型形參實(shí)例化的參數(shù)稱為模板實(shí)參,用模板實(shí)參實(shí)例化的函數(shù)稱為模板函數(shù)。模板函數(shù)的生成就是將函數(shù)模板的類型形參實(shí)例化的過程。例如:
使用中應(yīng)注意的幾個問題:
(1)函數(shù)模板允許使用多個類型參數(shù),但在template定義部分的每個形參前必須有關(guān)鍵字typename或class,即:
template<class 數(shù)據(jù)類型參數(shù)標(biāo)識符1,…,class 數(shù)據(jù)類型參數(shù)標(biāo)識符n>
<返回類型><函數(shù)名>(參數(shù)表)
{
函數(shù)體
}
template<class T>
int I;
T min(T x,T y)
{
函數(shù)體
}
(3)模板函數(shù)類似于重載函數(shù),但兩者有很大區(qū)別:函數(shù)重載時,每個函數(shù)體內(nèi)可以執(zhí)行不同的動作,但同一個函數(shù)模板實(shí)例化后的模板函數(shù)都必須執(zhí)行相同的動作。
2 函數(shù)模板的異常處理
函數(shù)模板中的模板形參可實(shí)例化為各種類型,但當(dāng)實(shí)例化模板形參的各模板實(shí)參之間不完全一致時,就可能發(fā)生錯誤,如:
template<typename T>
void min(T &x, T &y)
{ return (x<y)?x:y; }
void func(int i, char j)
{
min(i, i);
min(j, j);
min(i, j);
min(j, i);
}
例子中的后兩個調(diào)用是錯誤的,出現(xiàn)錯誤的原因是,在調(diào)用時,編譯器按最先遇到的實(shí)參的類型隱含地生成一個模板函數(shù),并用它對所有模板函數(shù)進(jìn)行一致性檢查,例如對語句
min(i, j);
先遇到的實(shí)參i是整型的,編譯器就將模板形參解釋為整型,此后出現(xiàn)的模板實(shí)參j不能解釋為整型而產(chǎn)生錯誤,此時沒有隱含的類型轉(zhuǎn)換功能。解決此種異常的方法有兩種:
(1)采用強(qiáng)制類型轉(zhuǎn)換,如將語句min(i, j);改寫為min(i,int( j));
(2)用非模板函數(shù)重載函數(shù)模板
方法有兩種:
(1)借用函數(shù)模板的函數(shù)體
此時只聲明非模板函數(shù)的原型,它的函數(shù)體借用函數(shù)模板的函數(shù)體。如改寫上面的例子如下:
template<typename T>
void min(T &x, T &y)
{ return (x<y)?x:y; }
int min(int,int);
void func(int i, char j)
{
min(i, i);
min(j, j);
min(i, j);
min(j, i);
}
執(zhí)行該程序就不會出錯了,因?yàn)橹剌d函數(shù)支持?jǐn)?shù)據(jù)間的隱式類型轉(zhuǎn)換。
(2)重新定義函數(shù)體
就像一般的重載函數(shù)一樣,重新定義一個完整的非模板函數(shù),它所帶的參數(shù)可以隨意。C++中,函數(shù)模板與同名的非模板函數(shù)重載時,應(yīng)遵循下列調(diào)用原則:
• 尋找一個參數(shù)完全匹配的函數(shù),若找到就調(diào)用它。若參數(shù)完全匹配的函數(shù)多于一個,則這個調(diào)用是一個錯誤的調(diào)用。
• 尋找一個函數(shù)模板,若找到就將其實(shí)例化生成一個匹配的模板函數(shù)并調(diào)用它。
• 若上面兩條都失敗,則使用函數(shù)重載的方法,通過類型轉(zhuǎn)換產(chǎn)生參數(shù)匹配,若找到就調(diào)用它。
•若上面三條都失敗,還沒有找都匹配的函數(shù),則這個調(diào)用是一個錯誤的調(diào)用。
3.小例子
#include <iostream>
#include <string>
using namespace std;
template<class T> void swap(T *x, T *y){//函數(shù)的模板,只要能使用等號賦值類型都可以用這個模板互換
T temp = *x;
*x = *y;
*y = temp;
}
void main(){
int i = 9, j = 5;
float k = 9.2, l = 5.6;
swap(&i, &j);//交換整形
swap(&k, &l);//交換浮點(diǎn)型
cout<<"i = "<<i<<" , j = "<<j<<endl;
cout<<"k = "<<k<<" , l = "<<l<<endl;
}
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery


