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

關(guān)于C++菱形運算符深度解析

 更新時間:2024年04月30日 09:07:13   作者:apocelipes  
從語言標(biāo)準(zhǔn)來說,c++里沒有什么菱形運算符,c++20里雖然新增了一個運算符operator<=>,但這個和所謂的菱形運算符沒有任何關(guān)系,下面通過本文探討C++里也有菱形運算符嗎這一問題探討,感興趣的朋友跟隨小編一起看看吧

最近在翻《c++函數(shù)式編程》的時候看到有一小節(jié)在說c++14新增了“菱形運算符”。我尋思c++里好像沒什么運算符叫這名字啊,而且c++14新增的功能很少,我也不記得有添加這種語法特性。一瞬間我有些懷疑我的記憶了,所以為了查漏補缺,我寫了這篇文章。

什么是菱形運算符

這個概念在Java里比較多見:

List<String> myList = new ArrayList<>();

這東西在Java里的學(xué)名是diamond operator,表示使用泛型類并且類型參數(shù)在左側(cè)的表達(dá)式已給出因此在右側(cè)可以省略。

簡單的說就是讓你少寫幾次重復(fù)的類型參數(shù)。因為看起來像個菱形所以得名菱形運算符。

然后我們偶爾會在c++里看到形狀上很相似的東西:

std::sort(vec.begin(), vec.end(), std::greater<>());

<>出現(xiàn)在模板的特化中是我們所熟悉的,但這個std::greater<>()是什么呢?

c++沒有菱形運算符

先說結(jié)論,從語言標(biāo)準(zhǔn)來說,c++里沒有什么菱形運算符。

c++20里雖然新增了一個運算符operator<=>,但這個和所謂的菱形運算符沒有任何關(guān)系。

那問題來了,std::greater<>()是什么以及為什么書里說是c++14新增的特性呢?難道書里瞎說的嗎?但事實是這樣的示例代碼在c++14以及之后的標(biāo)準(zhǔn)下可以正常編譯運行,而且這本書的質(zhì)量尚可,雖然會在措辭上犯些小錯(比如c++沒有菱形運算符)但不至于花大篇幅去胡說八道。

當(dāng)然,要想回答這個問題我們得先復(fù)習(xí)點基礎(chǔ)知識。

<>在c++里的作用

先說結(jié)論,在c++里看到<>,絕大多數(shù)都是在為模板提供類型參數(shù),當(dāng)然這種東西我們不討論:(a<1, 2>b),這里<>是在兩個不同的表達(dá)式里。

那既然用來提供類型參數(shù),那為什么可以啥都不提供呢?答案是有兩類情況確實可以。

第一類是在函數(shù)模板上,類型參數(shù)可以自動推導(dǎo)時:

template <typename T>
void f(const T&)
{
    std::cout << "f<T>\n";
}
template <>
void f(const int&)
{
    std::cout << "f<int>\n";
}
void f(const int&)
{
    std::cout << "f\n";
}
int main()
{
    f(1);    // f
    f<>(1);  // f<int>
    f(1.2);  // f<T>
}

非模板函數(shù)在重載決議中的優(yōu)先級總是高于模板的,因此f(1)這樣的表達(dá)式總是會用到最下面定義的那個非模板函數(shù)f。這時候我們可以用f<int>(1)來直接調(diào)用函數(shù)模板f,而函數(shù)模板的類型參數(shù)如果能從參數(shù)推導(dǎo)出來的話,可以不明確給出(也就是后面的f(1.2)那樣的),而在我們現(xiàn)在這句表達(dá)式里,我們既要明確使用函數(shù)模板,又想讓類型參數(shù)被自動推導(dǎo),就得使用f<>(1)。

另一種情況不分類模板還是函數(shù)模板,當(dāng)模板的類型參數(shù)有默認(rèn)值時,可以靠<>來使用這些默認(rèn)值:

template <typename T = void>
struct Wrapper
{
    using wrappered = T;
};
// Wrapper<> 等于 Wrapper<void>
static_assert(std::is_same_v<Wrapper<>::wrappered, Wrapper<void>::wrappered>);

在第二種情況下,因為沒顯示給出類型參數(shù),且這里沒法使用類型推導(dǎo),因此編譯器使用了類型參數(shù)的默認(rèn)值,這里是void。

觀察比較仔細(xì)的話其實會發(fā)現(xiàn)上面兩種情況其實是一件事,<>相當(dāng)于沒有顯示給出任何類型參數(shù),于是對這些沒有顯示指定的類型參數(shù),編譯器會先嘗試類型推導(dǎo),如果沒法推導(dǎo)則會檢查這些類型參數(shù)是否有默認(rèn)值,有就利用默認(rèn)值。如果上面這兩步都沒法得到能正常使用的類型參數(shù),模板會被SFINAE淘汰或者報出編譯錯誤。

這并不是什么新語法,是從有模板開始就一直存在的規(guī)則。

現(xiàn)在我們可以看看std::greater<>()是什么了,首先std::greater是個類模板,然后它接受一個類型參數(shù),這個參數(shù)在c++14之后有了默認(rèn)值void,因此std::greater<>()std::greater<void>()。

c++14中究竟添加了什么

既然c++14并沒有添加“菱形運算符”,那究竟新增了什么呢?

在已經(jīng)知道了std::greater<>()的真身后,找起來就很容易了,所以我很快找到了對應(yīng)的新特性:n3421

這個特性是這樣的:原先我們要用標(biāo)準(zhǔn)庫提供的謂詞模板,需要自己指明參數(shù)類型,這樣寫起來很麻煩而且對于那種嵌套的或者元素類型復(fù)雜的容器來說寫明參數(shù)類型不僅費時而且費力,更要命的是對于map,一不小心是會有性能問題的:

for_each(map.begin(), map.end(), std::pred<std::pair<std::string, int64_t>>());

上述代碼的問題在于正確的參數(shù)類型應(yīng)該是std::pair<const std::string, int64_t>,我們漏掉了const,這會導(dǎo)致pair整個被復(fù)制一遍,性能是無比底下的。要徹底避免這種錯誤,就得利用自動類型推導(dǎo)。

然而前面說了,標(biāo)準(zhǔn)庫提供的謂詞基本全是類模板,類模板的模板參數(shù)要么依賴默認(rèn)值要么得顯示指定,怎么才能依賴自動推導(dǎo)呢。

于是這個新特性最精彩的地方來了:原先的模板的調(diào)用運算符不是模板參數(shù)也是定死的,但我們可以新加一個默認(rèn)參數(shù),然后針對這個默認(rèn)參數(shù)的類型進(jìn)行完全特化,在特化里提供一個泛型的operator(),這樣就能利用函數(shù)模板來自動推導(dǎo)參數(shù)類型了,而且以前的代碼不受影響。

默認(rèn)參數(shù)的設(shè)置也是有講究的,需要用一個謂詞用不到的且不會影響老代碼的類型,運氣不錯,void正好符合條件(void上幾乎沒法做什么操作,因此也不會被指定給這些謂詞做類型參數(shù)),因此現(xiàn)在的greater的代碼是下面這樣的:

// 注意默認(rèn)值是void
template <typename T = void> struct greater {
    constexpr bool operator()(const T& lhs, const T& rhs) const 
    {
        return lhs > rhs;
    }
};
// 針對greater<void>的完全特化
template <> struct greater<void> {
    template <class T, class U> auto operator()(T&& t, U&& u) const
    -> decltype(std::forward<T>(t) > std::forward<U>(u))
    { return std::forward<T>(t) > std::forward<U>(u); }
};

當(dāng)使用std::greater<T>()的時候,代碼的邏輯和原來一樣,當(dāng)使用std::greater<void>()的時候,返回的Functor的函數(shù)調(diào)用運算符是個模板,可以自己推導(dǎo)參數(shù)類型和返回值類型。至于為啥greater<void>的內(nèi)部構(gòu)造可以和其他情況實例化的greater區(qū)別這么大,這個是c++的特性:模板的不同實例之間是可以異構(gòu)的。

而且因為類型參數(shù)的默認(rèn)值就是void,因此可以簡寫成std::greater<>()。

所以c++14只是給標(biāo)準(zhǔn)庫里可以代替運算符的模板們增加了默認(rèn)類型參數(shù)和一個泛型的調(diào)用運算符,利用這些可以簡化代碼并確保類型安全。

真相是其實沒啥菱形運算符,只是利用了以前就存在的模板的特性簡化了標(biāo)準(zhǔn)庫的使用,讓人少寫點字。達(dá)成的效果倒是和Java的菱形運算符差不多。

總結(jié)

顯然書里有夸大成分,老話說盡信書不如無書,還得小心檢驗才是。

順便我們復(fù)習(xí)了現(xiàn)代c++的重要原則:能依賴自動類型推導(dǎo)的地方,沒必要自己手寫。

因此應(yīng)該多寫這樣的代碼:std::sort(vec.begin(), vec.end(), std::greater<>());

不過還有最后一個問題,為啥不直接用lambda呢?那是因為能指定類型參數(shù)的泛型lambda要在c++20才出現(xiàn),在這之前想要讓lambda完全做到類型安全得費點功夫,而且lambda整體上也不如直接用標(biāo)準(zhǔn)庫提供的std::greater<>()std::less<>()之類的簡潔易懂。

到此這篇關(guān)于C++里的菱形運算符的文章就介紹到這了,更多相關(guān)C++菱形運算符內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java?C++?leetcode面試零矩陣

    Java?C++?leetcode面試零矩陣

    這篇文章主要為大家介紹了Java?C++題解leetcode面試零矩陣示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-10-10
  • C++11新特性std::make_tuple的使用

    C++11新特性std::make_tuple的使用

    這篇文章主要介紹了C++11新特性std::make_tuple的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • 深入解析C中的數(shù)值與真假

    深入解析C中的數(shù)值與真假

    本篇文章是對C中數(shù)值與真假進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-05-05
  • VC使用編譯時間作為版本號標(biāo)識的方法

    VC使用編譯時間作為版本號標(biāo)識的方法

    這篇文章主要介紹了VC使用編譯時間作為版本號標(biāo)識的方法,需要的朋友可以參考下
    2017-03-03
  • C語言單鏈表貪吃蛇小游戲

    C語言單鏈表貪吃蛇小游戲

    這篇文章主要為大家詳細(xì)介紹了C語言單鏈表貪吃蛇小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-02-02
  • C++中STL的優(yōu)先隊列priority_queue詳解

    C++中STL的優(yōu)先隊列priority_queue詳解

    這篇文章主要介紹了C++中STL的優(yōu)先隊列priority_queue詳解,今天講一講優(yōu)先隊列(priority_queue),實際上,它的本質(zhì)就是一個heap,我從STL中扒出了它的實現(xiàn)代碼,需要的朋友可以參考下
    2023-08-08
  • Qt自定義控件實現(xiàn)儀表盤

    Qt自定義控件實現(xiàn)儀表盤

    這篇文章主要為大家詳細(xì)介紹了Qt如何自定義控件實現(xiàn)儀表盤,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • C++實現(xiàn)歌手比賽評分系統(tǒng)

    C++實現(xiàn)歌手比賽評分系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C++實現(xiàn)歌手比賽評分系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • C語言中回調(diào)函數(shù)和qsort函數(shù)的用法詳解

    C語言中回調(diào)函數(shù)和qsort函數(shù)的用法詳解

    這篇文章主要為大家詳細(xì)介紹一下C語言中回調(diào)函數(shù)和qsort函數(shù)的用法教程,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)C語言有一定幫助,需要的可以參考一下
    2022-07-07
  • 詳解C++之類和對象(2)

    詳解C++之類和對象(2)

    類是創(chuàng)建對象的模板,一個類可以創(chuàng)建多個對象,每個對象都是類類型的一個變量;創(chuàng)建對象的過程也叫類的實例化。每個對象都是類的一個具體實例(Instance),擁有類的成員變量和成員函數(shù)
    2021-11-11

最新評論