【C++精華鋪】8.C++模板初階

目錄

1. 泛型編程

2. 函數模板

2.1 函數模板的概念及格式

?2.2 函數模板的原理

?2.3 模板的實例化

2.4 模板參數的匹配原則

?3. 類模板

3.1 類模板格式

3.2 類模板的實例化


1. 泛型編程

? ? ? ? 什么是泛型編程?泛型編程是避免使用某種具體類型而去使用某種通用類型來進行程序編寫的方式,依次來復用某段代碼而避免大規模功能相似重復冗余的代碼。下面的代碼如果想用泛型編程該如何實現。

int add(int a,int b)
{return a + b;
}
double add(double a, int b)
{return a + b;
}
double add(int a, double b)
{return a + b;
}

????????馬云有次說了這么一句話:“世界是懶人創造的,懶不是傻懶,如果你想少干,就要想出懶的方法。 要懶出風格,懶出境界。”C++必然也是有風格有境界的,所以C++中設計了模板實現了泛型編程。

2. 函數模板

2.1 函數模板的概念及格式

? ? ? ? 模板就是一種模具,通過給這個模具中放不同的材料(類型),來獲得不同材料的產品,以此來提高我們的工作效率。而函數模板就是某個函數的模具,與類型無關,在使用的時候參數化,在我們給出特定類型就會生成特定類型的版本。

? ? ? ? 函數模板的格式:

template<typename T1, typename T2 ...typename Tn>
返回值類型 函數名(形參列表)
{
?? ?//函數體
}

? ? ? ? ?根據這個函數的模板我們就可以實現上述add函數的模板:

template<typename L, typename R,typename RET>
RET add(L l, R r)
{return l + r;
}

注:typename是用來定義模板參數的關鍵字,也可以用class代替。

?2.2 函數模板的原理

? ? ? ? 函數模板本身不是函數,而是一個模具,當我們傳入實際類型的時候編譯器會根據我們傳入的實參類型來推演生成對應類型的函數進行調用。

?2.3 模板的實例化

? ? ? ? 當我們將類型傳入模板來生產函數的時候,稱之為模板的實例化,模板的實例化分為隱式實例化和顯式實例化。

????????1. 隱式實例化:隱式實例化就是我們不去指定類型,讓編譯根據我們傳入的實參判斷其類型傳入模板參數列表進行實例化。如下:

template<typename L, typename R>
int add(L l, R r)
{return l + r;
}
int main()
{int a = 0, b = 1;cout << add(a, b) << endl;  //根據我傳入的a和b自動判斷類型進行模板實例化。
}

? ? ? ? 細心的同學發現,這里我將原來的代碼中?typename RET?刪掉了,原因就是模板不會對函數的返回值類型進行自動判斷,而需要我們手動指定(也就是顯式實例化)。

????????2. 顯式實例化:顯式實例化就是我們手動向模板傳遞類型,然后由編譯器進行實例化。顯式實例化的格式為:

函數名<類型1,類型2,...>(實參列表);

????????如:?

template<typename L, typename R, typename RET>
RET add(L l, R r)
{return l + r;
}
int main()
{int a = 0, b = 1;cout << add<int,int,int>(a, b) << endl;
}

? ? ? ? 上述代碼我們傳進去的 a和b的類型與模板參數類型是匹配的,如果不匹配的話編譯器會進行隱式類型轉化,如果轉化失敗就報錯。如下:
? ? ? ? 類型轉換成功:

template<typename L, typename R, typename RET>
RET add(L l, R r)
{return l + r;
}
int main()
{double a = 0.1;int b = 1;cout << add<int,int,int>(a, b) << endl;  //a:double類型隱式轉換成int類型
}

輸出:

? ? ? ? 類型轉化失敗:?

template<typename L, typename R, typename RET>
RET add(L l, R r)
{return l + r;
}class Date
{//
};
int main()
{int a = 0, b = 1;cout << add<int,Date,int>(a, b) << endl;  //a:類型轉化失敗
}

輸出:

E0304?? ?沒有與參數列表匹配的 函數模板 "add" 實例

2.4 模板參數的匹配原則

? ? ? ? 一個函數的模板函數可以與非模板函數同時存在,并且在沒有顯式實例化且類型匹配的情況下會優先匹配非模板函數,如果是顯式實例化才會調用模板,如下。

template<class T>
void swap(T& a, T& b)
{std::swap(a, b);std::cout << "我是模板 ";std::cout << a << ' ' << b << std::endl;}
void swap(int& a, int& b)
{std::swap(a, b);std::cout << "我是非模板 ";std::cout << a << ' ' << b << std::endl;}
int main()
{int a = 1, b = 2;std::cout << a << ' ' << b << std::endl;swap(a, b);   //類型匹配優先調用非模板函數swap<int>(a, b);  //調用模板實例化
}

輸出:

? ? ? ? 當我們沒有去寫模板的時候,函數發生類型不匹配可能會進行隱式類型轉換,但是有了模板之后就不會發生隱式類型轉化,而是使用模板實例化出來一個更為合適的函數。如下:
? ? ? ? 沒有模板時會發生隱式實例化:

void test(int a, int b)
{std::cout << "我不是模板" << std::endl;
}int main()
{test(2, 2.0);//2.0發生隱式類型轉化成int;
}

輸出:

? ? ? ? 存在模板不會發生隱式類型轉化,而是使用模板實例化出來一個更為合適的函數:

template<class T1,class T2>
void test(T1 a, T2 b)
{std::cout << "我是模板" << std::endl;
}
void test(int a, int b)
{std::cout << "我不是模板" << std::endl;
}
int main()
{test(2, 2.0);//不發生隱式類型轉化而是隱式實例化模板
}

輸出:

?3. 類模板

3.1 類模板格式

? ? ? ? 和函數模板一樣,類模板是類的一個模具,類模板格式如下:

template<class T1,class T2,class T3,...class Tn>
class 類模板名
{
?? ?//類體
};

? ? ? ? 我們普通類是可以聲明和定義分離的,如果類模板要實現聲明和定義分離,那么在定義的時候也要加上模板聲明。如:

template<class T1, class T2>
class A
{~A();  //聲明
};template<class T1,class T2> 
A<T1, T2>::~A()   //定義
{//...
}

3.2 類模板的實例化

?????????類模板的實例化和函數模板相同,需要在類名后加上尖括號并且指定類型。注:模板名不是類,實例化后才是一個類。

template<class T>
class Date
{//..
};
int main()
{Date<int>;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/40603.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/40603.shtml
英文地址,請注明出處:http://en.pswp.cn/news/40603.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

mysql中INSERT INTO ... ON DUPLICATE KEY UPDATE的用法,以及與REPLACE INTO 語句用法的異同

INSERT INTO ... ON DUPLICATE KEY UPDATE 是 MySQL 中一種用于插入數據并處理重復鍵沖突的語法。與之相似的還有 REPLACE INTO 語句。以下是它們的用法和異同點的詳細說明&#xff1a; 一、INSERT INTO ... ON DUPLICATE KEY UPDATE INSERT INTO ... ON DUPLICATE KEY UPDAT…

NET域名的優勢

NET域名是互聯網上最常見的頂級域名之一&#xff0c;其開放使用日期遠比其他主要頂級域名早&#xff0c;始于1985年。其作為商業網絡服務提供者的域名&#xff0c;主要用于企業、組織和個人等在網絡上建立自己的網站。本文將從以下三個方面介紹NET域名。 一、NET域名的歷史 N…

帶你了解—使用內網穿透,公網遠程訪問本地硬盤文件

文章目錄 前言1. 下載cpolar和Everything軟件3. 設定http服務器端口4. 進入cpolar的設置5. 生成公網連到本地內網穿透數據隧道 總結 前言 隨著云概念的流行&#xff0c;不少企業采用云存儲技術來保存辦公文件&#xff0c;同時&#xff0c;很多個人用戶也感受到云存儲帶來的便利…

如何使用Java代碼收集網站所有功能

使用Java代碼收集網站所有功能的步驟可以這么實現: 1. 使用JSoup等工具解析網站首頁HTML,獲取超鏈接、表單等元素。 Document doc JSoup.connect("http://website.com").get(); Elements links doc.select("a[href]"); Elements forms doc.select(&qu…

學習ts(四)聯合類型、交叉類型、類型斷言

聯合類型 使用聯合類型定義屬性和方法&#xff0c;只要符合其中一種即可 let myPhone: string | number 010-7788 // let myPhone1: string | number true 因為沒有包含boolean值 會報錯const fn (something: number | boolean): boolean > {return !!something }con…

【CSS動畫01--登錄】

CSS動畫01--登錄 介紹代碼HTMLCSSJS 介紹 當鼠標不同方向的劃過時展示不同效果的登錄&#xff0c;以上是一個簡單的圖片展示 代碼 HTML <!DOCTYPE html> <html> <head><meta http-equiv"content-type" content"text/html; charsetutf-8&…

生物筆記——暑期學習筆記(四)

生物筆記——暑期學習筆記&#xff08;四&#xff09; 文章目錄 前言一、R篇1. unname()2. duplicated()3. 數據提取4. 分組 二、生信篇1. 文本處理常用命令2. 命令輸出1. 重定向2. 多命令執行 3. 文本工具4. 本地hmm鑒定1. hmmer軟件安裝2. 文件準備3. 基于hmm的鑒定 總結 前言…

【制作npm包5】npm包制作完整教程,我的第一個npm包

制作npm包目錄 本文是系列文章&#xff0c; 作者一個橙子pro&#xff0c;本系列文章大綱如下。轉載或者商業修改必須注明文章出處 一、申請npm賬號、個人包和組織包區別 二、了解 package.json 相關配置 三、 了解 tsconfig.json 相關配置 四、 api-extractor 學習 五、npm包…

MySQL的配置文件my.cnf與my.ini

一、my.cnf與my.ini win系統&#xff0c;MySQL配置文件為my.ini 其他系統&#xff08;Ubuntu、CentOS、macOS)MySQL配置文件為my.cnf 二、my.cnf與my.ini的路徑 2.1 默認路徑 MySQL 的配置文件 my.cnf 可能位于多個位置&#xff0c;具體取決于安裝方式和操作系統。以下是一…

Redis如何處理內存溢出的情況?

當Redis的內存使用達到上限時&#xff0c;會出現內存溢出的情況。Redis提供了幾種處理內存溢出的機制&#xff1a; 內存淘汰策略&#xff1a;Redis提供了多種內存淘汰策略&#xff0c;用于在內存不足時選擇要移除的鍵。常見的淘汰策略包括&#xff1a; LRU&#xff08;Least Re…

2023年國賽數學建模思路 - 案例:最短時間生產計劃安排

文章目錄 0 賽題思路1 模型描述2 實例2.1 問題描述2.2 數學模型2.2.1 模型流程2.2.2 符號約定2.2.3 求解模型 2.3 相關代碼2.4 模型求解結果 建模資料 0 賽題思路 &#xff08;賽題出來以后第一時間在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 最短時…

Kotlin開發筆記:使用委托進行拓展

Kotlin開發筆記&#xff1a;使用委托進行拓展 導言 在OO語言(面向對象)中&#xff0c;我們經常會用到委托或者代理的思想。委托和代理在乍一看很相似&#xff0c;其實其各有各的側重點&#xff0c;這里我引用ChatGpt的回答&#xff1a; 委托&#xff08;Delegation&#xff09…

探索C語言中的常見排序算法

探索C語言中的常見排序算法 排序算法是計算機科學中至關重要的基礎知識之一&#xff0c;它們能夠幫助我們對數據進行有序排列&#xff0c;從而更高效地進行搜索、插入和刪除操作。在本篇博客中&#xff0c;我們將深入探討C語言中的一些常見排序算法&#xff0c;包括它們的工作…

在C中使用Socket實現多線程異步TCP消息發送

目錄 基礎知識開始實現主要函數說明結束語 在本篇文章中&#xff0c;我們會探討如何在C語言中使用socket來實現多線程&#xff0c;異步發送TCP消息的系統。雖然C標準庫并沒有原生支持異步和多線程編程&#xff0c;但是我們可以結合使用POSIX線程&#xff08;pthread&#xff09…

Java解決四大查找(一)

Java解決四大查找 一.線性查找1.1 題目1.2 思路分析1.3 代碼演示 二.二分查找(雙指針法)2.1 題目2.2 思路分析(圖解加文字)2.3 代碼演示 一.線性查找 1.1 題目 在數組{1&#xff0c;8&#xff0c;1024&#xff0c;521&#xff0c;1889}中查找數字8&#xff0c;如果有&#xff…

【知識分享】高防服務器的防御機制

【知識分享】高防服務器的防御機制 易受到攻擊的網站選擇接入高防服務更安全&#xff0c;大家對于這個都清楚!但是對于高防服務如何實現防御來保障安全的&#xff0c;又了解多少呢?今天壹基比小源&#xff08;貳伍壹叁壹叁壹貳玖捌&#xff09;就來說說高防服務實現防御的常規…

地址解析協議-ARP

ARP協議 無論網絡層使用何種協議&#xff0c;在實際網絡的鏈路上傳輸數據幀時&#xff0c;最終必須使用硬件地址 地址解析協議&#xff08;Address Resolution Protocol&#xff0c;ARP&#xff09;&#xff1a;完成IP地址到MAC地址的映射&#xff0c;每個主機都有一個ARP高速緩…

【數據結構】二叉樹篇| 綱領思路02+刷題

博主簡介&#xff1a;努力學習的22級計算機科學與技術本科生一枚&#x1f338;博主主頁&#xff1a; 是瑤瑤子啦每日一言&#x1f33c;: 所謂自由&#xff0c;不是隨心所欲&#xff0c;而是自我主宰。——康德 目錄 一、前言二、刷題1、翻轉二叉樹 2、二叉樹的層序遍歷?3、 二…

線性代數再回顧

最近&#xff0c;在深度學習線性代數&#xff0c;之前大一的時候學過線性代數&#xff0c;但那純屬于是應試用的&#xff0c;考試一考完&#xff0c;啥都忘了&#xff0c;也說出不出個所以然&#xff0c;所以&#xff0c;在B站的MIT的線性代數以及3blue1brown線性代數的本質中去…

git命令使用

君子拙于不知己,而信于知己。——司馬遷 清屏&#xff1a;clear 查看當前面板的路徑&#xff1a;pwd 查看當前面板的文件&#xff1a;ls 創建文件夾&#xff1a;mkdir 文件夾名 創建文件&#xff1a;touch 文件名 刪除文件夾&#xff1a;rm -rf 文件夾名 刪除文件&#xff1a;r…