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

歡迎訪問 生活随笔!

生活随笔

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

c/c++

[c++][语言语法]函数模板和模板函数 及参数类型的运行时判断

發布時間:2024/4/15 c/c++ 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 [c++][语言语法]函数模板和模板函数 及参数类型的运行时判断 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

參考:http://blog.csdn.net/beyondhaven/article/details/4204345

參考:http://blog.csdn.net/joeblackzqq/article/details/7460704

1.函數模板的聲明和模板函數的生成

?

1.1函數模板的聲明

函數模板可以用來創建一個通用的函數,以支持多種不同的形參,避免重載函數的函數體重復設計。它的最大特點是把函數使用的數據類型作為參數。

函數模板的聲明形式為:

template<typename 數據類型參數標識符>

<返回類型><函數名>(參數表)

{

??? 函數體

}

其中,template是定義模板函數的關鍵字;template后面的尖括號不能省略;typename(或class)是聲明數據類型參數標識符的關鍵字,用以說明它后面的標識符是數據類型標識符。這樣,在以后定義的這個函數中,凡希望根據實參數據類型來確定數據類型的變量,都可以用數據類型參數標識符來說明,從而使這個變量可以適應不同的數據類型。例如:

template<typename T>

T fuc(T x, int y)

{

??? T x;

??? //……

}

如果主調函數中有以下語句:

double d;

int a;

fuc(d,a);

則系統將用實參d的數據類型double去代替函數模板中的T生成函數:

double fuc(double x,int y)

{

??? double x;

??? //……

}

函數模板只是聲明了一個函數的描述即模板,不是一個可以直接執行的函數,只有根據實際情況用實參的數據類型代替類型參數標識符之后,才能產生真正的函數。

關鍵字typename也可以使用關鍵字class,這時數據類型參數標識符就可以使用所有的C++數據類型。

1.2.模板函數的生成

函數模板的數據類型參數標識符實際上是一個類型形參,在使用函數模板時,要將這個形參實例化為確定的數據類型。將類型形參實例化的參數稱為模板實參,用模板實參實例化的函數稱為模板函數。模板函數的生成就是將函數模板的類型形參實例化的過程。例如:

使用中應注意的幾個問題:

⑴ 函數模板允許使用多個類型參數,但在template定義部分的每個形參前必須有關鍵字typename或class,即:

template<class 數據類型參數標識符1,…,class 數據類型參數標識符n>

<返回類型><函數名>(參數表)

{

???? 函數體

}

⑵ 在template語句與函數模板定義語句<返回類型>之間不允許有別的語句。如下面的聲明是錯誤的:

template<class T>

int I;

T min(T x,T y)

{

?? 函數體

}

⑶ 模板函數類似于重載函數,但兩者有很大區別:函數重載時,每個函數體內可以執行不同的動作,但同一個函數模板實例化后的模板函數都必須執行相同的動作。

?

2 函數模板的異常處理

?

函數模板中的模板形參可實例化為各種類型,但當實例化模板形參的各模板實參之間不完全一致時,就可能發生錯誤,如:

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);

}

例子中的后兩個調用是錯誤的,出現錯誤的原因是,在調用時,編譯器按最先遇到的實參的類型隱含地生成一個模板函數,并用它對所有模板函數進行一致性檢查,例如對語句

min(i, j);

先遇到的實參i是整型的,編譯器就將模板形參解釋為整型,此后出現的模板實參j不能解釋為整型而產生錯誤,此時沒有隱含的類型轉換功能。解決此種異常的方法有兩種:

⑴采用強制類型轉換,如將語句min(i, j);改寫為min(i,int( j));

⑵用非模板函數重載函數模板

方法有兩種:

① 借用函數模板的函數體

此時只聲明非模板函數的原型,它的函數體借用函數模板的函數體。如改寫上面的例子如下:

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);

}

執行該程序就不會出錯了,因為重載函數支持數據間的隱式類型轉換。

② 重新定義函數體

就像一般的重載函數一樣,重新定義一個完整的非模板函數,它所帶的參數可以隨意。C++中,函數模板與同名的非模板函數重載時,應遵循下列調用原則:

? 尋找一個參數完全匹配的函數,若找到就調用它。若參數完全匹配的函數多于一個,則這個調用是一個錯誤的調用。

? 尋找一個函數模板,若找到就將其實例化生成一個匹配的模板函數并調用它。

? 若上面兩條都失敗,則使用函數重載的方法,通過類型轉換產生參數匹配,若找到就調用它。

?若上面三條都失敗,還沒有找都匹配的函數,則這個調用是一個錯誤的調用。

?

c++類模板及參數類型的運行時判斷

  • /*?
  • ????C++類模板及參數類型的運行時判斷(typeid)?
  • */??
  • ??
  • #include?<stdio.h>??
  • #include?<typeinfo>??
  • #include?<vector>??
  • using?namespace?std;??
  • ??
  • template<class?T>??
  • class?Exercise??
  • {??
  • ????public:??
  • ????????typedef?T?Type;??
  • ????????typedef?vector?<?Type?>?VT;??
  • ????????Exercise(int?n);??
  • ????????void?Display();??
  • ????private:??
  • ????????VT?dv;??
  • };??
  • ??
  • template<class?T>??
  • Exercise<T>::Exercise(int?n)??
  • {??
  • ????T?v;??
  • ????printf("type:?%s\n",?typeid(T).name());??
  • ????for(int?i?=?0;?i?<?n;?i++)??
  • ????{??
  • ????????v?=?1.1?*?(i+1);??
  • ????????dv.push_back(v);??
  • ????}??
  • }??
  • ??
  • template<class?T>??
  • void?Exercise<T>::Display()??
  • {??
  • ????char?fmt[2][10]?=?{"%d\t",?"%.2f\t"};??
  • ????char?*p?=?fmt[0];??
  • ????if(typeid(T)?==?typeid(double)?||?typeid(T)?==?typeid(float))??
  • ????????p?=?fmt[1];??
  • ????for(typename?vector<T>::iterator?it?=?dv.begin();?it?!=?dv.end();?it++)??
  • ????{??
  • ????????printf(p,?*it);??
  • ????}??
  • ????printf("\n\n");??
  • }??
  • ??
  • int?main()??
  • {??
  • ????Exercise<double>?ex1(5);??
  • ????ex1.Display();??
  • ??
  • ????Exercise<float>?ex2(5);??
  • ????ex2.Display();??
  • ??
  • ????Exercise<int>?ex3(5);??
  • ????ex3.Display();??
  • ??????
  • ????Exercise<long>?ex4(5);??
  • ????ex4.Display();??
  • ??
  • ??
  • ????printf("\n");??
  • ????printf("%s\n",?typeid(ex1).name());??
  • ????printf("%s\n",?typeid(ex2).name());??
  • ????printf("%s\n",?typeid(ex3).name());??
  • ????printf("%s\n",?typeid(ex4).name());??
  • ??
  • ????return?0;??
  • }??
  • ?

    轉載于:https://www.cnblogs.com/lyggqm/p/5291097.html

    總結

    以上是生活随笔為你收集整理的[c++][语言语法]函数模板和模板函数 及参数类型的运行时判断的全部內容,希望文章能夠幫你解決所遇到的問題。

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