深入理解c++20 concepts
concepts在c++20中被引入,其作用是對模板參數(shù)進(jìn)行約束,極大地增強(qiáng)了c++模板的功能。
在c++20之前,如果希望獲取類似的效果,使用起來并不方便。
沒有concept時(shí),如何實(shí)現(xiàn)對模板參數(shù)進(jìn)行約束?
static_assert
我們可以使用static_assert去對模板類型T進(jìn)行約束。如下所示:
#include <type_traits> #include <iostream> template<class T> void test(T a) { ? ? static_assert(std::is_integral<T>()); ? ? std::cout << "T is integral" << std::endl; } int main() { ? ? test(10); ? ? test<double>(12.3);? }
但是該種方法不太好,因?yàn)樾枰獙tatic_assert嵌入到函數(shù)的內(nèi)部,這意味著即使類型不對,模板還是成功的匹配上了,只是在模板函數(shù)內(nèi)部展開時(shí)出現(xiàn)編譯錯(cuò)誤。
SFINAE
SFINAE是Substitution Failure Is Not An Error的縮寫,翻譯過來的意思是替換失敗并不是一個(gè)錯(cuò)誤。
SFINAE是模板元編程中常見的一種技巧,如果模板實(shí)例化后的某個(gè)模板函數(shù)(模板類)對該調(diào)用無效,那么將繼續(xù)尋找其他重載決議,而不是引發(fā)一個(gè)編譯錯(cuò)誤。
因此一句話概括SFINAE,就是模板匹配過程中會(huì)嘗試各個(gè)重載模板,直到所有模板都匹配失敗,才會(huì)認(rèn)為是真正的錯(cuò)誤。
例如下面這個(gè)經(jīng)典的例子:
struct Test { ? ? ?typedef int foo; }; template <typename T> void f(typename T::foo) {} ?// Definition #1 template <typename T> void f(T) {} ?// Definition #2 int main() { ? ? f<Test>(10); ?// Call #1. ? ? f<int>(10); ? // Call #2. Without error (even though there is no int::foo) ? ? ? ? ? ? ? ? // thanks to SFINAE. }
f<Test>(10)最終將使用到第一個(gè)模板定義, 而f<int>(10)最終將使用到第二個(gè)模板定義。
SFINAE 原則最初是應(yīng)用于上述的模板編譯過程。后來被C++開發(fā)者發(fā)現(xiàn)可以用于做編譯期的決策,配合sizeof可以進(jìn)行一些判斷:類是否定義了某個(gè)內(nèi)嵌類型、類是否包含某個(gè)成員函數(shù)等。例如STL中迭代器中的has_iterator_category。
template <typename T> struct has_iterator_category { ? ? struct two { char a; char b; }; ? ? template <typename C> ? ? static two& test(typename C::iterator_category*); ? ? template <typename> ? ? static char& test(...); ? ? static const bool value = sizeof(test<T>(nullptr)) == sizeof(two); };
enable_if
enable_if的出現(xiàn)使得SFINAE使用上更加方便,進(jìn)一步擴(kuò)展了上面has_xxx,is_xxx的作用。而enable_if實(shí)現(xiàn)上也是使用了SFINAE。
enable_if的定義簡單, 即當(dāng)_Test是true時(shí),將不會(huì)有type的類型定義,而當(dāng)_Test是false時(shí),將會(huì)有type的類型定義。
// STRUCT TEMPLATE enable_if template <bool _Test, class _Ty = void> struct enable_if {}; // no member "type" when !_Test template <class _Ty> struct enable_if<true, _Ty> { // type is _Ty for _Test ? ? using type = _Ty; };
下面是利用enable_if去實(shí)現(xiàn)SFINAE的方式。
#include <type_traits> #include <iostream> template <typename T> typename std::enable_if<std::is_integral<T>::value, T>::type? test(T value) { ? ? std::cout<<"T is intergal"<<std::endl; ? ? return value; } template <typename T> typename std::enable_if<!std::is_integral<T>::value, T>::type? test(T value) { ? ? std::cout<<"T is not intergal"<<std::endl; ? ? return value; } int main() { ? ? test(100); ? ? test('a'); ? ? test(100.1); }
可以看到SFINAE的主要實(shí)現(xiàn)了is_xxx和has_xxx的語義,但是其語法并不簡單,對使用者有較高要求,并且即便寫正確了,可讀性也相對較差。
有了concept之后如何使用?
聲明concept
聲明一個(gè)concept的語法如下所示:
template < template-parameter-list > concept concept-name = constraint-expression;
例如,約束T是一個(gè)整形:
template <typename T> concept integral = std::is_integral_v<T>;
也可以使用requires更加靈活地定義concept,例如下面的例子要求T類型擁有一個(gè)名字叫做print的方法,另外需要擁有一個(gè)toString方法,并且返回值是string類型。
template <typename T> concept printable = requires(T t) { t.print(); //1 {t.toString()} -> std::same_as<std::string>; //2 };
使用concept
使用concept有三種方式:
方法1:直接將concept嵌入模板的類型的尖括號(hào)<>內(nèi)
template<Arithmetic T> void f(T a){/*function definition*/}; we can enforce it just after template declaration using requires:
方法2:在模板聲明的下方使用requires關(guān)鍵字
template<class T> requires Arithmetic<T> void f(T a) {/*function definition*/}; or after the function declaration
方法3:使用后置形式,直接在函數(shù)聲明的后方使用requires關(guān)鍵字添加約束。
#include<concepts> template<class T> void f(T a) requires integral<T> // integral is in header <concepts> {/*function definition*/};
此外,concept還可以使用邏輯運(yùn)算符 && 和 ||。例如:
template <class T> concept Integral = std::is_integral<T>::value; template <class T> concept SignedIntegral = Integral<T> && std::is_signed<T>::value; template <class T> concept UnsignedIntegral = Integral<T> && !SignedIntegral<T>;
下面是兩個(gè)完整的例子來看看concepts是如何實(shí)現(xiàn)了is_xxx和has_xxx的功能。
第一個(gè)例子中test模板T只能是整形或者是double。實(shí)現(xiàn)了is_xxx。
#include <iostream> template <class T> concept Integral = std::is_integral<T>::value; template <class T> concept IsDouble = std::is_same<T, double>::value; template<Integral T> void test(T a) { ? ? std::cout << "test(int) functin called" << std::endl; } template<IsDouble T> void test(T a) { ? ? std::cout << "test(double) functin called" << std::endl; } int main() { ? ? test(100); ? ? test(10.0); }
第二個(gè)例子中print函數(shù)要求t擁有print函數(shù)。實(shí)現(xiàn)了has_xxx。
#include <iostream> template <typename T> concept Has_print = requires(T t) { ? ? t.print(); //1 }; class HasPrint { public: ? ? void print(){}; }; class NoPrint { }; template<Has_print T> void print(T t) { ? ? t.print(); } int main() { ? ? HasPrint t; ? ? print(t); ? ? NoPrint t2; ? ? print(t2); //將報(bào)編譯錯(cuò)誤 }
總結(jié)
c++20的concepts增強(qiáng)了模板對于參數(shù)類型約束的功能,語法簡單,可以提高代碼的可讀性。
如果你的項(xiàng)目不能使用較新的標(biāo)準(zhǔn),那么還是要老老實(shí)實(shí)的使用SNINAE,enable_if那一套東西。
到此這篇關(guān)于深入理解c++20 concepts的文章就介紹到這了,更多相關(guān)c++20 concepts內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12Window10下安裝VS2022社區(qū)版的實(shí)現(xiàn)步驟(圖文教程)
很多和同學(xué)們在接觸c語言的時(shí)候都是使用VS,本文主要介紹了Window10下如何安裝VS2022社區(qū)版的實(shí)現(xiàn)步驟,具有一定的參考價(jià)值,感興趣的可以了解一下2024-02-02QT+Quick實(shí)現(xiàn)自定義組件的示例詳解
這篇文章主要為大家詳細(xì)介紹了如何利用QT+Quick實(shí)現(xiàn)自定義組件(按鈕、下拉菜單等),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-11-11C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)堆排序圖文示例
這篇文章主要為大家介紹了C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)堆排序的圖文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05