www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁(yè) > > 充電吧
[導(dǎo)讀]C++11的模板類(lèi)型判斷——std::is_same和std::decay問(wèn)題提出:有一個(gè)模板函數(shù),函數(shù)在處理int型和double型時(shí)需要進(jìn)行特殊的處理,那么怎么在編譯期知道傳入的參數(shù)的數(shù)據(jù)類(lèi)型是i

C++11的模板類(lèi)型判斷——std::is_same和std::decay

問(wèn)題提出:有一個(gè)模板函數(shù),函數(shù)在處理int型和double型時(shí)需要進(jìn)行特殊的處理,那么怎么在編譯期知道傳入的參數(shù)的數(shù)據(jù)類(lèi)型是int型還是double型呢?
如:

#includetemplatevoid?typeCheck(TYPE?data)
{
????//do?something?check?data?type
????//std::cout<<?out?put?the?type
}

這里就需要用到C++11的type_traits頭文件了,type_traits頭文件定義了很多類(lèi)型檢查相關(guān)的方法,上面的例子具體用到了其中兩個(gè)結(jié)構(gòu):

std::is_same 判斷類(lèi)型是否一致

位于頭文件中

這個(gè)結(jié)構(gòu)體作用很簡(jiǎn)單,就是兩個(gè)一樣的類(lèi)型會(huì)返回true

bool?isInt?=?std::is_same::value;?//為true

下面是官方的例子:

#include#include#includevoid?print_separator()
{
????std::cout?<<?"-----n";
}

int?main()
{
????std::cout?<<?std::boolalpha;

????std::cout?<<?std::is_same::value?<<?'n';???//?true
????std::cout?<<?std::is_same::value?<<?'n';???//?false
????std::cout?<<?std::is_same::value?<<?'n';?//?false

????print_separator();

????std::cout?<<?std::is_same::value?<<?"n";??????????//?true
????std::cout?<<?std::is_same::value?<<?"n";?//?false
????std::cout?<<?std::is_same::value?<<?"n";???//?true

????print_separator();

????//?unlike?other?types?'char'?is?not?'unsigned'?and?not?'signed'
????std::cout?<<?std::is_same::value?<<?"n";??????????//?true
????std::cout?<<?std::is_same::value?<<?"n";?//?false
????std::cout?<<?std::is_same::value?<<?"n";???//?false
}

通過(guò)std::is_same即可判斷兩個(gè)類(lèi)型是否一樣,特別在模板里面,在不清楚模板的參數(shù)時(shí),此功能可以對(duì)一些特定的參數(shù)類(lèi)型進(jìn)行特殊的處理。

這里說(shuō)個(gè)題外話(huà),大家是否通過(guò)std::is_same發(fā)現(xiàn),char既不是unsigned char也不是signed char,char就是char,這和int是signed int的縮寫(xiě)是不一樣的,char的表達(dá)范圍可能等同于signed char,也可能等同于unsigned char,取決于編譯器,一般是等同于signed char,但這個(gè)僅僅是范圍等同,就像32位上int和long范圍是一樣的,但不是同一個(gè)類(lèi)型。

因?yàn)橛猛静煌?,char用于表達(dá)字符,理論上不應(yīng)該關(guān)心其正負(fù)的實(shí)現(xiàn),而signed char 和 unsigned char 用于表達(dá)數(shù)值,或可移植的char。

回到正文,std::is_same可以判斷兩種類(lèi)似是否一樣,那么用在模板里就是利器了,本位一開(kāi)始提到的那個(gè)問(wèn)題就可以這樣寫(xiě):

#includetemplatetypeCheck(TYPE?data)
{
????if(std::is_same::value)
????{
????????std::cout<<"int?type";
????????//do?something?int?
????}
????else
????{
????????//.........
????}
}

視乎很美好,再看一個(gè)示例:

//?is_same?example
#include#include#includetypedef?int?integer_type;
struct?A?{?int?x,y;?};
struct?B?{?int?x,y;?};
typedef?A?C;

int?main()?{
??????std::cout?<<?std::boolalpha;
??????std::cout?<<?"is_same:"?<<?std::endl;
??????std::cout?<<?"int,?const?int:?"?<<?std::is_same::value?<<?std::endl;//false
??????std::cout?<<?"int,?int&:?"?<<?std::is_same::value?<<?std::endl;//false
??????std::cout?<<?"int,?const?int&:?"?<<?std::is_same::value?<<?std::endl;//false
??????std::cout?<<?"int,?integer_type:?"?<<?std::is_same::value?<<?std::endl;//true
??????std::cout?<<?"A,?B:?"?<<?std::is_same::value?<<?std::endl;//false
??????std::cout?<<?"A,?C:?"?<<?std::is_same::value?<<?std::endl;//true
??????std::cout?<<?"signed?char,?std::int8_t:?"?<<?std::is_same::value?<<?std::endl;//true
??????return?0;
}

輸出:

is_same:
int,?const?int:?false
int,?int&:?false
int,?const?int&:?false
int,?integer_type:?true
A,?B:?false
A,?C:?true
signed?char,?std::int8_t:?true

發(fā)現(xiàn)std::is_same的判斷是很?chē)?yán)格的
但是有時(shí)候在編輯模板的時(shí)候又發(fā)現(xiàn)用std::is_same的判斷太過(guò)嚴(yán)格,還是之前的例子:

#include#include#includetemplatevoid?typeCheck(TYPE?data);

int?_tmain(int?argc,?_TCHAR*?argv[])
{
????int?a?=?1;
????const?int&?b?=?a;
????int&?c?=?a;
????int?d[12];
????const?int&?e?=?d[7];
????typeCheck(a);//int?type
????typeCheck(b);//int?type
????typeCheck(c);//int?type
????typeCheck(d[7]);//int?type
????typeCheck(e);//int?type
????typeCheck(8);//int?type
????system("pause");
????return?0;
}

templatevoid?typeCheck(TYPE?data)
{
????if(std::is_same::value)
????{
????????std::cout<<"int?type"<<std::endl;
????}
????else?if(std::is_same::value)
????{
????????std::cout<<"string?type"<<std::endl;
????}
????else
????{
????????std::cout<<"other?type";
????}
}

輸出:

int?type
int?type
int?type
int?type
int?type
int?type

測(cè)試后發(fā)現(xiàn),雖然變量b,c使用引用,但std::is_same還是能識(shí)別出來(lái)的,但是?。?
如果我顯示的指定模板參數(shù)類(lèi)型時(shí)情況有不一樣了:

#include#include#includetemplatevoid?typeCheck(TYPE?data);

int?_tmain(int?argc,?_TCHAR*?argv[])
{
????int?a?=?1;
????const?int&?b?=?a;
????int&?c?=?a;
????int?d[12];

????typeCheck(a);????????//int?type
????typeCheck(b);//other?type
????typeCheck(c);????????//other?type
????typeCheck(d[7]);//other?type
????typeCheck(8);????????????????//int?type
????system("pause");
????return?0;
}

templatevoid?typeCheck(TYPE?data)
{
????if(std::is_same::value)
????{
????????std::cout<<"int?type"<<std::endl;
????}
????else?if(std::is_same::value)
????{
????????std::cout<<"string?type"<<std::endl;
????}
????else
????{
????????std::cout<<"other?type";
????}
}

輸出:

int?type
other?type
other?type
other?type
int?type

瞬間結(jié)果就不一樣了,這很好了解,從上面可知道,std::is_same對(duì)int和const intint &const int&等都是區(qū)別對(duì)待的,但在寫(xiě)模板函數(shù)時(shí),經(jīng)常會(huì)強(qiáng)制指定常引用進(jìn)行傳參,以免進(jìn)行數(shù)據(jù)拷貝,這時(shí)候is_same就做出了不相等的判斷,但是有時(shí)候其實(shí)我們還是希望TYPE和const TYPE& 是能認(rèn)為是一樣的,這時(shí)就需要std::decay進(jìn)行退化處理

std::decay 退化類(lèi)型的修飾

std::decay就是對(duì)一個(gè)類(lèi)型進(jìn)行退化處理,他的實(shí)現(xiàn)如下:

template<?class?T?>
struct?decay?{
private:
????typedef?typename?std::remove_reference::type?U;
public:
????typedef?typename?std::conditional<?
????????std::is_array::value,
????????typename?std::remove_extent::type*,
????????typename?std::conditional<?
????????????std::is_function::value,
????????????typename?std::add_pointer::type,
????????????typename?std::remove_cv::type
????????>::type
????>::type?type;
};

看著比較抽象,其實(shí)就是把各種引用啊什么的修飾去掉,把cosnt int&退化為int,這樣就能通過(guò)std::is_same正確識(shí)別出加了引用的類(lèi)型了
上面的例子改為:

#include?"stdafx.h"
#include#include#includetemplatevoid?typeCheck(TYPE?data);

int?_tmain(int?argc,?_TCHAR*?argv[])
{
????int?a?=?1;
????const?int&?b?=?a;
????int&?c?=?a;
????int?d[12];

????typeCheck(a);//int?type
????typeCheck(b);//int?type
????typeCheck(c);//int?type
????typeCheck(d[7]);//int?type
????typeCheck(8);//int?type
????system("pause");
????return?0;
}

templatevoid?typeCheck(TYPE?data)
{
????if(std::is_same<typename?std::decay::type,int>::value)
????{
????????std::cout<<"int?type"<<std::endl;
????}
????else
????{
????????std::cout<<"other?type"<<std::endl;
????}
}

在cppref有個(gè)更加詳細(xì)的例子:

#include#includetemplatestruct?decay_equiv?:?
????std::is_same<typename?std::decay::type,?U>::type?
{};

int?main()
{
????std::cout?<<?std::boolalpha
??????????????<<?decay_equiv::value?<<?'n'
??????????????<<?decay_equiv::value?<<?'n'
??????????????<<?decay_equiv::value?<<?'n'
??????????????<<?decay_equiv::value?<<?'n'
??????????????<<?decay_equiv::value?<<?'n'
??????????????<<?decay_equiv::value?<<?'n';
}

輸出:

true
true
true
true
true
true

總結(jié): 在模板里可以通過(guò)std::is_same判斷模板的類(lèi)型,從而實(shí)現(xiàn)對(duì)不同類(lèi)型的區(qū)別對(duì)待 在堆類(lèi)型要求不是非常嚴(yán)格的情況下,可以使用std::decay把類(lèi)型退化為基本形態(tài),結(jié)合std::is_same用,可以判斷出更多的情況

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀(guān)點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專(zhuān)欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

其實(shí)在 c++語(yǔ)言里面const修飾的才算是一個(gè)真正的常量,在 c 語(yǔ)言中 const 可以說(shuō)是個(gè)“冒牌貨”。為什么會(huì)這樣?其實(shí)是 c++ 編譯器對(duì) const 進(jìn)行了加強(qiáng),當(dāng) c++ 編譯器遇到常量聲明時(shí),不會(huì)像 c...

關(guān)鍵字: c++ C語(yǔ)言 const

返回函數(shù)的引用去初始化一個(gè)新的引用這個(gè)和前面一樣,都是不會(huì)產(chǎn)生副本,但是現(xiàn)在是用返回值去初始化一個(gè)引用聲明c,也就是說(shuō)這時(shí)候變成了變量temp的別名,在c的生命周期內(nèi)temp是一直有效的,這樣做完全可以。

關(guān)鍵字: c++ 返回值 引用聲明

C++是一種面向?qū)ο蟮母呒?jí)程序設(shè)計(jì)語(yǔ)言,是C語(yǔ)言的超集。

關(guān)鍵字: c++ C語(yǔ)言

c++程序員面試過(guò)程中基本上都會(huì)被問(wèn)到c++11新特性吧,你是怎么回答的呢? 本文基本上涵蓋了c++11的所有新特性,并有詳細(xì)代碼介紹其用法,對(duì)關(guān)鍵知識(shí)點(diǎn)做了深入分析,對(duì)重要的知識(shí)點(diǎn)我單獨(dú)寫(xiě)了相關(guān)文章并附上了相關(guān)鏈接,我...

關(guān)鍵字: c++11

很多人談到c++,說(shuō)它特別難,可能有一部分就是因?yàn)閏++的內(nèi)存管理吧,不像java那樣有虛擬機(jī)動(dòng)態(tài)的管理內(nèi)存,在程序運(yùn)行過(guò)程中可能就會(huì)出現(xiàn)內(nèi)存泄漏,然而這種問(wèn)題其實(shí)都可以通過(guò)c++11引入的智能指針來(lái)解決,相反我還認(rèn)為這...

關(guān)鍵字: c++11

c++11關(guān)于并發(fā)引入了好多好東西,這里按照如下順序介紹: std::thread相關(guān) std::mutex相關(guān) std::lock相關(guān) std::atomic相關(guān) std::call_once相關(guān) volatile相關(guān)...

關(guān)鍵字: 線(xiàn)程 c++11

分析:這是Adobe 公司2007 年校園招聘的最新筆試題。這道題除了考察應(yīng)聘者的C++ 基本功底外,還能考察反應(yīng)能力,是一道很好的題目。 在Java 中定義了關(guān)鍵字final ,被final 修飾的

關(guān)鍵字: c++ class

泛型算法中的定制操作很多算法都會(huì)比較輸入序列中的元素,通過(guò)定制比較動(dòng)作,可以控制算法按照編程者的意圖工作。本文以string排序?yàn)槔M(jìn)行說(shuō)明,首先是缺省的排序動(dòng)作:?vector v{"This","

關(guān)鍵字: c++

為什么是lambda?講了這么多天的lambda表達(dá)式,有一個(gè)很基本的問(wèn)題沒(méi)有回答:為什么叫l(wèi)ambda表達(dá)式呢?首先這個(gè)lambda就是羅馬字母λ,lambda表達(dá)式即λ表達(dá)式。數(shù)學(xué)上有一個(gè)概念叫λ

關(guān)鍵字: c++

? ? ? ? 假設(shè)我們有個(gè)函數(shù)用來(lái)揭示處理程序的優(yōu)先權(quán),另一個(gè)函數(shù)用來(lái)在某動(dòng)態(tài)分配所得的Widget 上進(jìn)行某些帶有優(yōu)先權(quán)的處理:int priority () ; void processWi

關(guān)鍵字: c++ effective
關(guān)閉