C++精要分析decltype的作用及用法
獲取表達(dá)式的類型
在編寫程序的過程中,我們可能會(huì)有一種需求,就是希望可以根據(jù)一個(gè)變量的類型,來定義具有相同類型的變量。例如定義int x = 0;,那么我們是否可以不使用int關(guān)鍵字,僅使用x就定義一個(gè)新的整型變量y呢?
答案是可以的,C++11新增的decltype關(guān)鍵字就是干這個(gè)用的。上述需求用代碼實(shí)現(xiàn)如下:
int x = 0; decltype(x) y = 2; // y的類型為int
decltype是在編譯期用來推導(dǎo)表達(dá)式類型的。其語法格式為:decltype(expression)。大家可以看到,decltype是可以對(duì)一個(gè)表達(dá)式取類型的,并不僅是單個(gè)的變量。所以,把形式再擴(kuò)展一下:
int x = 0; decltype(x) y = 2; decltype(x + y) z = 3; // z的類型為int
到這一步,相信大家已經(jīng)可以基本掌握其特性,在工作中能運(yùn)用了。當(dāng)然,僅知道這些還是不夠的,作為C++程序員怎么能停下探索的腳步呢。
推導(dǎo)規(guī)則
decltype的推導(dǎo)規(guī)則,是面試中最容易挖坑的地方。你要是不信,那就先回答下面這些問題吧:
const int func_one();
decltype(func_one()) a1 = 0; // a1是什么類型?
struct TestData { int x;};
cosnt TestData b_node = TestData();
decltype(b_node.x) b1 = 0; // b1是什么類型?
decltype((b_node.x)) b2 = b1; // b2是什么類型?
int n = 0, m = 0;
decltype(n + m) c1 = 0; // c1是什么類型?
decltype(n += m) c2 = c1; // c2是什么類型?注釋中有五個(gè)問題,如果你全都答對(duì)而且不是蒙的,那請開班授課吧,我會(huì)第一個(gè)報(bào)名。先公布一下答案,看看自己答對(duì)了多少吧。
a1: int
b1: int
b2: const int &
c1: int
c2: int &
如果這個(gè)答案讓你覺得有些暈頭轉(zhuǎn)向,不要緊,先來看下規(guī)則描述吧:
- 如果expression表達(dá)式是標(biāo)識(shí)符、類訪問表達(dá)式,
decltype(exp)和exp的類型一致; - 如果expression是函數(shù)調(diào)用,則
decltype(exp)和返回值的類型一致; - 其他情況,如果expression是一個(gè)左值,則
decltype(exp)是exp類型的左值引用,否則和exp類型一致。
現(xiàn)在,將規(guī)則理解之后,再看一遍代碼和答案,是否找到規(guī)律了呢?相信在面試中遇到這樣的問題,你已經(jīng)可以應(yīng)對(duì)自如了。
返回類型后置
在說明decltype的一個(gè)高級(jí)用法之前,我們先了解C++11的一個(gè)新特性,就是函數(shù)返回類型后置。與之相對(duì)的,就是返回類型前置,這是我們最熟悉的函數(shù)聲明格式。例如:int foo();
而返回類型后置的示例如下:
auto foo() -> int {
return 0;
}
在上面的代碼中,auto關(guān)鍵字是一個(gè)占位符,int是其實(shí)際返回類型。初看起來,后置聲明與前置聲明在功能上是一樣的,那它難道是一個(gè)多余的設(shè)計(jì)嗎?它當(dāng)然自有用武之地。
在需要返回比較復(fù)雜的類型時(shí),使用后置式聲明可以簡化代碼并使其可讀性更好。例如要返回的類型是函數(shù)指針,前置式聲明就必須先用typedef進(jìn)行預(yù)定義,否則語法不允許。而后置式聲明則可以直接實(shí)現(xiàn),無需預(yù)定義,如下代碼所示。
int exam(bool b) {
int ret = -1;
if (b) {
ret = 0;
}
else if (!b) {
ret = 1;
}
return ret;
}
auto foo() -> int(*)(bool) {
return exam;
}
int main() {
auto fn = foo();
cout << fn(true) << endl;
cout << fn(false) << endl;
}
高級(jí)用法
現(xiàn)在正式介紹decltype與函數(shù)返回類型后置相結(jié)合,在模板編程中的用法,就是用于推導(dǎo)函數(shù)模板的返回類型。之所以將此歸為高級(jí)用法,也是因?yàn)槟0逶贑++中雖然功能強(qiáng)大,但屬實(shí)復(fù)雜不易理解。一般是編寫基礎(chǔ)功能庫或是算法庫時(shí),使用到模板的特性。
先看一段示例代碼:
template<class T1, class T2>
auto sum(T1 t1, T2 t2) -> decltype(t1 + t2) {
return t1 + t2;
}
int main() {
auto ret = sum(4.6, 123);
cout << ret << endl;
}
其精髓之處,就在于可以靈活支持T1與T2不同類型的組合,而不必為每種返回類型都去寫一個(gè)實(shí)現(xiàn)。例如int+int, double+int, string+string等各種組合情況。
但如果是把decltype(t1 + t2)以前置寫法替換auto,則會(huì)產(chǎn)生編譯錯(cuò)誤。道理很簡單,編譯器對(duì)t1+t2的參數(shù)類型還一無所知,只有在解析到返回值時(shí),才能最終確定函數(shù)的返回類型,這就是decltype加上函數(shù)返回類型后置在模板編程中的妙用。
到此這篇關(guān)于C++精要分析decltype的作用及用法的文章就介紹到這了,更多相關(guān)C++ decltype內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言詳解如何實(shí)現(xiàn)帶頭雙向循環(huán)鏈表
帶頭雙向循環(huán)鏈表:結(jié)構(gòu)最復(fù)雜,一般用在單獨(dú)存儲(chǔ)數(shù)據(jù)。實(shí)際中使用的鏈表數(shù)據(jù)結(jié)構(gòu),都是帶頭雙向循環(huán)鏈表。另外這個(gè)結(jié)構(gòu)雖然結(jié)構(gòu)復(fù)雜,但是使用代碼實(shí)現(xiàn)以后會(huì)發(fā)現(xiàn)結(jié)構(gòu)會(huì)帶來很多優(yōu)勢,實(shí)現(xiàn)反而簡單2022-04-04
解析sizeof, strlen, 指針以及數(shù)組作為函數(shù)參數(shù)的應(yīng)用
本篇文章是對(duì)sizeof, strlen, 指針以及數(shù)組作為函數(shù)參數(shù)的應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
VC++實(shí)現(xiàn)輸出GIF到窗體并顯示GIF動(dòng)畫的方法
這篇文章主要介紹了VC++實(shí)現(xiàn)輸出GIF到窗體并顯示GIF動(dòng)畫的方法,需要的朋友可以參考下2014-07-07
Opencv下載和導(dǎo)入Visual studio2022的實(shí)現(xiàn)步驟
本文主要介紹了Opencv下載和導(dǎo)入Visual studio2022的實(shí)現(xiàn)步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05
C++利用easyx圖形庫實(shí)現(xiàn)創(chuàng)意天天酷跑小游戲
這篇文章主要為大家詳細(xì)介紹了C++如何利用easyx圖形庫實(shí)現(xiàn)創(chuàng)意小游戲——天天酷跑,文中的示例代碼講解詳細(xì),快跟隨小編一起了解一下吧2023-03-03
makefile如何調(diào)用靜態(tài)庫的方法實(shí)現(xiàn)
這篇文章主要介紹了makefile如何調(diào)用靜態(tài)庫的方法實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12

