C++函數模板和類模板

来源:https://www.cnblogs.com/qlzstudy/archive/2022/07/27/16526938.html
-Advertisement-
Play Games

C++語言全盤繼承了C語言的標準庫,其中包換非常豐富的系統函數,例如輸入/輸出函數、數學函數、字元串處理函數和動態記憶體分配函數等。C++語言另外又增加了一些新的庫,我們把C++語言新增的這部分庫稱為C++標準庫。C++語言的模板技術包括函數模板和類模板。模板技術是一種代碼重用技術,函數和類是C++語 ...


  C++語言全盤繼承了C語言的標準庫,其中包換非常豐富的系統函數,例如輸入/輸出函數、數學函數、字元串處理函數和動態記憶體分配函數等。C++語言另外又增加了一些新的庫,我們把C++語言新增的這部分庫稱為C++標準庫。C++語言的模板技術包括函數模板和類模板。模板技術是一種代碼重用技術,函數和類是C++語言中兩種主要的重用代碼形式。代碼模板使源程式更加凝練。

一、函數模板

  函數模板的基本原理是通過數據類型的參數化,將一組演算法相同但所處理數據類型不同的重載函數凝練成一個函數模板。

1、函數模板的定義和使用

  函數模板的定義語法形式:

  template <類型參數列表>

  函數類型 函數名(形參參數列表)

  {

    函數體

  }

函數模板的幾點語法細則:

  • 定義函數模板以關鍵字template開頭
  • 類型參數列表可以聲明一個或多個類型參數,每個類型參數以"typename 類型參數名"或"class 類型參數名"的形式聲明,類型參數之間用逗號”,“隔開。
  • 函數模板的其餘部分,包括函數類型、函數名、形式參數列表以及函數體,它們和普通函數的定義形式沒有什麼區別。
  • 使用typename和class聲明的類型參數可視為一種新的數據類型,可用來定義函數類型(即返回值類型)、形參類型和局部變數類型。類型參數是表示數據類型的參數,再實際調用時可被替換成任意數據類型

示例:

#include<iostream>
using namespace std;
template <typename T>  //定義函數模板Max,聲明一個類型參數T
T Max(T x,T y)   //使用類型參數T定義函數類型和形參x和y的類型
{
    return(x>y?x:y);
}
int main()
{
    cout << Max(5,10) <<endl; //調用返回最大值10
    cout << Max(5.2,10.2) <<endl; //調用返回最大值10.2
    return 0;
}

  函數模板可以像普通函數一樣被調用。再調用者看來,函數模板的類型參數像是一種通用數據類型。

2、函數模板的編譯原理

  函數模板是具有類型參數的函數。類型參數是表示數據類型的參數,可指代任意一種實際數據類型。編譯器再編譯到函數模板的調用語句時,根據位置對應關係從實參數據類型推導出類型參數所指代的數據類型,然後按照函數模板自動生成一個該類型的函數定義代碼。不同類型實參的函數模板調用語句將生成不同類型的重載函數。函數模板將數據類型參數化,調用時會呈現參數多態性

3、函數模板的聲明

  跟普通函數一樣函數模板也需要先定義後調用的原則。如果函數模板定義再調用後,或定義再其他程式文件中,則應先聲明後調用。函數模板的聲明語法形式為:

  template <類型參數列表>

  函數類型 函數名(形參參數列表)

  或

  template <類型參數列表> 函數類型 函數名(形參參數列表);

#include<iostream>
using namespace std;
template <typename T>  //函數模板Max定義聲明
T Max(T x,T y);
int main()
{
    cout << Max(5,10) <<endl; //函數模板Max使用
    cout << Max(5.2,10.2) <<endl; 
    return 0;
}
template <typename T>  //函數模板Max定義
T Max(T x,T y)   
{
    return(x>y?x:y);
}

  程式員再編程時可以靈活使用模板技術。在定義多個重載函數時可以考慮是否將他們定義成一個函數模板,這樣可以凝練函數代碼。再定義單個函數時也可以考慮定義成函數模板,這樣可以提高函數代碼的可重用性。對於調用函數模板的程式員而言,函數模板和普通函數沒有什麼區別。唯一不同的時函數模板可以處理不同類型的數據。

二、類模板

  應用模板技術,也可以將一組功能相同但所處理數據類型不同的類凝練成一個類模板。編譯時,再由編譯器按照類模板自動生成針對不同數據類型的類定義代碼。

1、類模板的定義和使用

  定義類模板的語法形式:

  template <類型參數列表>

  class 類名  //類聲明部分

  {

    類成員聲明

  }

  //類實現部分:所有類外定義的函數成員,必須按如下的語法形式將它們定義成函數模板

  template <類型參數列表>

  函數類型 類名<類型參數名列表>::函數名(形式參數列表)

  { 函數體 }

類模板的幾點語法細則:

  • 定義類模板以關鍵字template開頭
  • 類型參數列表可以聲明一個或多個類型參數,每個類型參數以"typename 類型參數名"或"class 類型參數名"的形式聲明,類型參數之間用逗號”,“隔開。
  • 類模板定義的其餘部分,包括類名、類成員聲明以及類實現部分,它們和普通類的定義形式基本相同。
  • 定義類模板的函數成員,如果再類內定義(內聯函數),其語法形式和普通類的函數成員沒區別;如果再類外(即類實現部分)定義,則必須按照函數模板的語法形式來定義,並且還要再函數名前面加“類名<類型參數名列表>::”限定。
  • 使用typename和class聲明的類型參數可視為一種新的數據類型,類型參數是表示數據類型的參數,再使用類模板時可被替換成任意數據類型。

定義好的類模板可以像普通類一樣被用來定義對象。使用類模板定義對象時,需要明確給出類模板中類型參數所指代的實際數據類型。其語法形式如下:類模板名 <實際數據類型列表> 對象名1,對象名2 .....;

類模板語法示例:

在類內定義函數成員(內聯) 在類外定義函數模板成員
#include<iostream>
using namespace std;
template <typename T> //類模板A
class A  //類聲明部分
{
    private: //聲明兩個私有數據成員
        T a1;
        int a2;
    public:
        A(T p1,int p2) //定義構造函數
        { a1 = p1; a2 = p2}
        void Show() //顯示數據成員
        { cout << a1 <<","<<a2 <<endl;}
        T Sum() //求數據成員的和
        {return (T)(a1+a2); }
};
//無類實現部分

int main()
{
    //用類模板定義對象
    A <double> o1(10.5,6); //double型對象
    o1.Show(); //顯示:10.5,6
    cout<<o1.Sum()<<endl;//顯示:16.5
    
    A <int> o2(10,6); //int型對象
    o2.Show(); //顯示:10,6
    cout<<o1.Sum()<<endl;//顯示:16
    return 0;
}
#include<iostream>
using namespace std;
template <typename T> //類模板A
class A  //類聲明部分
{
    private: //聲明兩個私有數據成員
        T a1;
        int a2;
    public:
        A(T p1,int p2); //定義構造函數
        void Show(); //顯示數據成員
        T Sum(); //求數據成員的和
};
//類實現部分
template <typename T>
A <T>::A(T p1,int p2) 
{ a1 = p1; a2 = p2}
template <typename T>
void A <T>::Show() 
{ cout << a1 <<","<<a2 <<endl;}
template <typename T>
T A <T>::Sum()
{return (T)(a1+a2); }
int main()
{
    //用類模板定義對象
    A <double> o1(10.5,6); //double型對象
    o1.Show(); //顯示:10.5,6
    cout<<o1.Sum()<<endl;//顯示:16.5
    
    A <int> o2(10,6); //int型對象
    o2.Show(); //顯示:10,6
    cout<<o1.Sum()<<endl;//顯示:16
    return 0;
}

2、類模板的編譯原理

  當定義到類模板定義對象語句時,編譯器將根據所給出的實際數據類型來取代類型參數T。例如 A <double> o1(10.5,6);  編譯時將類模板中類類型參數綁定到某個具體數據類型的過程,稱為類模板的實例化。實例化所生成的類稱為類模板的實例類。實例類是一個普通的類,可以用來定義對象。

  類模板編譯原理:類模板是具有類型參數的類。類型參數是表示數據類型的參數,可指代任意實際數據類型。編譯器在編譯到使用類模板定義對象語句時,將首先按照所給定的實際數據類型對類模板進行實例化,生成一個實例類。最終,編譯器使用實例類來定義所需要的對象。

3、類模板的繼承和派生

  類模板可以被繼承,派生出新類。以類模板為基類定義派生類,可以在派生時實例化,也可以繼續定義派生類模板。

1、定義實例化派生類

 定義實例化派生類就是在派生類繼承基類的時候將類型參數賦值,此時派生類對基類進行實例化。

實例化派生類示例
#include<iostream>
using namespace std;
template <typename T> //類模板基類Base
class Base  //類聲明部分
{
    private: //聲明私有數據成員
        T a;
    public:
        Base(T x) { a=x; }
        void Show() { cout<< "a="<<a<<","; }
};
//無類實現部分

class Derived:public Base<double> //公有繼承基類模板Base,派生時實例化
{
    private: //聲明新增數據成員
        int b;
    public:
        //註意派生類構造函數寫法
        Derived(double p1,int p2):Base<double>(p1) { b=p2; } 
        //新增函數成員Show
        void Show() { Base <double>::Show(); cout<< "b="<<b<<endl; }
};

int main()
{
    Derived obj(10.5,6);//定義派生類Derived對象obj
    obj.Show(); //顯示結果:a=10.5,b=6
    return 0;
}

  在編譯到派生類Derived的定義代碼時,編譯器將按照所給定的實際數據類型double對類模板Base進行實例化,生成一個double型的實例類,最終派生類Derived繼承的是該實例類。

2、定義派生類模板

  定義派生類模板是派生類在繼承類模板基類時不進行實例化,因此派生類仍然是一個類模板。

派生類模板示例
#include<iostream>
using namespace std;
template <typename T> //類模板基類Base
class Base  //類聲明部分
{
    private: //聲明私有數據成員
        T a;
    public:
        Base(T x) { a=x; }
        void Show() { cout<< "a="<<a<<","; }
};
//無類實現部分

//公有繼承類模板Base,派生類仍為類模板
template <typename T,typename TT> //新增類型參數TT
class Derived:public Base<T> //公有繼承基類模板Base,定義派生類模板
{
    private: //聲明新增數據成員
        TT b;
    public:
        //註意派生類構造函數寫法
        Derived(T p1,TT p2):Base<T>(p1) { b=p2; } 
        //新增函數成員Show
        void Show() { Base <T>::Show(); cout<< "b="<<b<<endl; }
};

int main()
{
    Derived<double,int> obj(10.5,6);//定義派生類Derived對象obj
    obj.Show(); //顯示結果:a=10.5,b=6
    return 0;
}

  和其他類模板一樣,派生類模板Derived在定義對象時需要明確給出派生類模板中類型參數所指代的實際數據類型。程式員編程時,在定義多個功能相同但處理數據類型不同的類時應考慮是否可以將它們合併成一個類模板,這樣可以凝練代碼。再定義單個類時也可以考慮升級成類模板,這樣可以提高函數代碼的可重用性。對於調用類模板的程式員而言,類模板和普通類沒有什麼區別。只是在使用類模板時需要給出類型參數所指代的實際數據類型。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、 Vue概述 Vue (讀音/vju/, 類似於view)是一套用於構建用戶界面的漸進式框架,發佈於2014年2月。 與其它大型框架不同的是,Vue被設計為可以自底向上逐層應用。 Vue的核心庫只關註視圖層,不僅易於上手,還便於與第三方庫(如: vue-router: 跳轉,vue-resour ...
  • vue項目導航菜單問題 目標:橫向菜單點擊跳轉,顏色變換,刷新可保持狀態 // 模板template中通過迴圈菜單列表生成,動態類名改變顏色 <li v-for="(item, index) in navList" :key="index" v-text="item.name" :class="{ ...
  • 1 WebRTC音視頻通話功能簡介 本文介紹如何基於WebRTC快速實現一個簡單的實時音視頻通話。 在開始之前,您可以先瞭解一些實時音視頻推拉流相關的基礎概念: 流:一組按指定編碼格式封裝的音視頻數據內容。一個流可以包含幾個軌道,比如視頻和音頻軌道。 推流:把採集階段封包好的音視頻數據流推送到 ZE ...
  • 首先,瞭解預解析之前先看兩個問題 1.大家思考下 這個結果會是多少呢? console.log(num); var num=10; 結果是 undefined 2.這個輸出結果又會是多少呢? fun(); var fun=function(){ console.log(22); } 顯然這個結果報錯 ...
  • 條件控制語句及表達式 運算符及表達式 1.()前面不能直接用++ console.log(++(a--)); //() 不能和++ 一起使用 2.str 與 Number值比較(字元串比較時會自動變為Number值) console.log('123A'>213);//false 自動轉為numbe ...
  • 在電腦發展的早期,一直都是集中式計算,計算能力依賴大型電腦。隨著互聯網的發展,繁重的業務需要巨大的計算能力才能完成,而集中式計算無法滿足要求,大型電腦的價格也非常昂貴。分散式計算將任務分解成更小的部分,分配給多台電腦處理,這樣可以節約整體計算時間,大大提高計算效率。互聯網大型網站往往面臨高並... ...
  • hello,大家好呀,我是小樓。今天不寫BUG,來聊一聊註冊中心。 標題本來想叫《如何設計一個註冊中心》,但網上已經有好多類似標題的文章了。所以打算另闢蹊徑,換個角度,如何組裝一個註冊中心。 組裝意味著不必從0開始造輪子,這也比較符合許多公司對待自研基礎組件的態度。 知道如何組裝一個註冊中心有什麼用 ...
  • 一、freemarker介紹 FreeMarker 是一款 模板引擎 即一種基於模板和要改變的數據, 並用來生成輸出文本(HTML網頁,電子郵件,配置文件,源代碼等)的通用工具。 它不是面向最終用戶的,而是一個Java類庫,是一款程式員可以嵌入他們所開發產品的組件 模板編寫為FreeMarker T ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...