欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入理解c++20 concepts

 更新時(shí)間:2023年06月25日 11:42:09   作者:程序員小x  
本文主要介紹了深入理解c++20 concepts,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

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)文章

  • VS+QT編譯環(huán)境中字符亂碼問題解決方法

    VS+QT編譯環(huán)境中字符亂碼問題解決方法

    編碼就是把?個(gè)字符編碼成二進(jìn)制碼存起來的方式,而解碼就是把這個(gè)二進(jìn)制碼按照原本編碼的規(guī)則還原成原來的字符,這篇文章主要介紹了VS+QT編譯環(huán)境中字符亂碼問題詳解,需要的朋友可以參考下
    2024-01-01
  • 淺析C++中的多態(tài)與文件操作

    淺析C++中的多態(tài)與文件操作

    多態(tài)是面向?qū)ο缶幊蹋∣OP)的核心概念之一,它允許對象在相同操作下表現(xiàn)出不同的行為,本文主要為大家介紹了C++中多態(tài)與文件操作的相關(guān)知識(shí),希望對大家有所幫助
    2024-04-04
  • 一文帶你學(xué)習(xí)C++中的派生機(jī)制

    一文帶你學(xué)習(xí)C++中的派生機(jī)制

    C++是一門面向?qū)ο蟮木幊陶Z言,其中的派生機(jī)制是其重要的面向?qū)ο筇匦灾?。本文我們就來詳?xì)地學(xué)習(xí)一下C++中的派生機(jī)制的相關(guān)知識(shí)吧
    2023-04-04
  • C++ Boost Parameter超詳細(xì)講解

    C++ Boost Parameter超詳細(xì)講解

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個(gè)可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-11-11
  • C++回溯算法中子集問題分析探討

    C++回溯算法中子集問題分析探討

    回溯法是一種選優(yōu)搜索法,按選優(yōu)條件向前搜索,以達(dá)到目標(biāo)。但當(dāng)探索到某一步時(shí),發(fā)現(xiàn)原先選擇并不優(yōu)或達(dá)不到目標(biāo),就退回一步重新選擇,這種走不通就退回再走的技術(shù)為回溯法,而滿足回溯條件的某個(gè)狀態(tài)的點(diǎn)稱為回溯點(diǎn)
    2023-03-03
  • C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng)

    C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • Window10下安裝VS2022社區(qū)版的實(shí)現(xiàn)步驟(圖文教程)

    Window10下安裝VS2022社區(qū)版的實(shí)現(xiàn)步驟(圖文教程)

    很多和同學(xué)們在接觸c語言的時(shí)候都是使用VS,本文主要介紹了Window10下如何安裝VS2022社區(qū)版的實(shí)現(xiàn)步驟,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-02-02
  • QT+Quick實(shí)現(xiàn)自定義組件的示例詳解

    QT+Quick實(shí)現(xiàn)自定義組件的示例詳解

    這篇文章主要為大家詳細(xì)介紹了如何利用QT+Quick實(shí)現(xiàn)自定義組件(按鈕、下拉菜單等),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2022-11-11
  • C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)堆排序圖文示例

    C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)堆排序圖文示例

    這篇文章主要為大家介紹了C語言植物大戰(zhàn)數(shù)據(jù)結(jié)構(gòu)堆排序的圖文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-05-05
  • C++超詳細(xì)講解引用和指針

    C++超詳細(xì)講解引用和指針

    引用是C++一個(gè)很重要的特性,顧名思義是某一個(gè)變量或?qū)ο蟮膭e名,對引用的操作與對其所綁定的變量或?qū)ο蟮牟僮魍耆葍r(jià),這篇文章主要給大家總結(jié)介紹了C++中引用的相關(guān)知識(shí)點(diǎn),需要的朋友可以參考下
    2022-06-06

最新評論