溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

c++17新特性是什么

發(fā)布時間:2021-01-29 13:21:12 來源:億速云 閱讀:639 作者:小新 欄目:編程語言

這篇文章主要介紹了c++17新特性是什么,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

c++17新特性有:1、對auto表達(dá)式推導(dǎo)的規(guī)則進(jìn)行了改變;2、lambda表達(dá)式可以捕獲“*this”;3、新增inline變量,可以直接將全局變量定義在頭文件中;4、條件表達(dá)式中支持初始化語句;5、枚舉的直接列表初始化等等。

C++17中的新特性

1. auto關(guān)鍵字

從c++11開始,auto關(guān)鍵字能夠通過初始化器推導(dǎo)出變量的類型。在c++14中,auto關(guān)鍵字的能力進(jìn)一步提升,能夠通過return語句推導(dǎo)出函數(shù)的返回類型。 使用auto關(guān)鍵字能夠提高編碼效率,同時能夠簡化重構(gòu)流程。但是,C++11中的auto推導(dǎo),往往結(jié)果與預(yù)期的不同。

c++11 中為了支持統(tǒng)一初始化,引入了新的統(tǒng)一初始化語法,如下所示。

// c++11
auto x3{ 1, 2 }; // std::initializer_list<int>
auto x4 = { 3 }; // decltype(x4) is std::initializer_list<int>
auto x5{ 3 };    // std::initializer_list<int>

這三種方式初始化的變量,最終類型推導(dǎo)的結(jié)果都是 std::initializer_list , 而不是我們認(rèn)為的int。 這是因為 當(dāng)用于auto聲明變量的表達(dá)式是{}括起來的,推導(dǎo)的型別就會變成 std::initializer_list。

在C++17中,對auto表達(dá)式推導(dǎo)的規(guī)則進(jìn)行了改變

// c++17
auto x3{ 1, 2 }; // error: not a single element
auto x4 = { 3 }; // decltype(x4) is std::initializer_list<int>
auto x5{ 3 };    // decltype(x5) is int

對比發(fā)現(xiàn), auto x5{3}, 會直接將變量推導(dǎo)成 x5, 而 x3{1, 2} 這種方式也會編譯失敗。auto推導(dǎo)的規(guī)則變得更加直觀。

2. lambda表達(dá)式

lambda也是c++11中引入的,在C++11中,lambda表達(dá)式只能用捕獲this,this是當(dāng)前對象的一個只讀的引用。 在C++17中,可以捕獲*this, *this是當(dāng)前對象的一個拷貝,捕獲當(dāng)前對象的拷貝,能夠確保當(dāng)前對象釋放后, lambda表達(dá)式能安全的調(diào)用this中的變量和方法。

3. inline變量

Inline 變量, inline變量可以讓變量有多于一次的定義。C++17之前,我們定義全局變量, 總需要將變量定義在cpp文件中,然后在通過extern關(guān)鍵字來告訴編譯器 這個變量已經(jīng)在其他地方定義過了。 inline變量出現(xiàn)后,我們可以直接將全局變量定義在頭文件中,而不用擔(dān)心出現(xiàn)redefine的錯誤信息。

4. 條件表達(dá)式中支持初始化語句

c++17中支持在 if 或者switch 語句中進(jìn)行初始化, 這個能力的出現(xiàn)能夠讓代碼更加的簡潔。

// c++17之前
map<int, string> c = {{1,"a"}};
{
    auto res = c.insert(make_pair(2, "b"));
    if(!res.second) {
        cout << "key 1 exist" << endl;
    } else {
        cout << "insert success, value:" << res.first->second << endl;
    }
}

上面的一段代碼,由于res是一個臨時變量,不想影響到后面的代碼,所以用一對花括號限制了其作用域。但是如果使用c++17的語法, 在if條件中初始化res,則代碼就會顯得更加簡潔

// c++17
map<int, string> c = {{1,"a"}};
if(auto res = c.insert(make_pair(2, "b")); !res.second ) {
    cout << "key 1 exist" << endl;
} else {
    cout << "insert success, value:" << res.first->second << endl;
}

c++17的標(biāo)準(zhǔn)庫也進(jìn)行了擴(kuò)充, 新增了下面幾種數(shù)據(jù)類型:

1. std::variant

std::variant是類型安全的聯(lián)合體,是一個加強(qiáng)版的 union,variant支持更加復(fù)雜的數(shù)據(jù)類型,例如map,string等等

2. std::optional

std::optional表示一個可能存在的值。 當(dāng)我們通過函數(shù)創(chuàng)建一個對象時,通常使用通過函數(shù)返回錯誤碼,而通過出參返回對象本身。 如果通過optional返回創(chuàng)建的實例,就會變得更加直觀,

std::optional 提供了下面幾個方法:

has_value()   // 檢查對象是否有值
value()       // 返回對象的值,值不存在時則拋出 std::bad_optional_access 異常
value_or()    // 值存在時返回值,不存在時返回默認(rèn)值

3. std::any

一個類型安全的可以保存任何值的容器

4. std::string_view

string_view我最早使用的是boost版本的,c++17中的string_view 和 boost類似。 string_view可以理解成原始字符串一個只讀引用。 string_view 本身沒有申請額外的內(nèi)存來存儲原始字符串的data, 僅僅保存了原始字符串地址和長度等信息。 在很多情況下,我們只是臨時處理字符串,本不需要對原始字符串的一份拷貝。 使用string_view可以減少不必要的內(nèi)存拷貝,可以提高程序性能。相比使用字符串指針,string_view做了更好的封裝。

需要注意的是,string_view 由于沒有原始字符串的所有權(quán),使用string_view 一定要注意原始字符串的生命周期。 當(dāng)原始的字符串已經(jīng)銷毀,則不能再調(diào)用string_view。

其他特性:

除此之外,C++17還增加了一些其他特性,文中沒有一一列出。

  • bool 表達(dá)式不能用 ++, – 這兩個自增(減)運(yùn)算符了

  • c++17中異常已經(jīng)成為了類型系統(tǒng)的一部分,

  • 枚舉的直接列表初始化

  • 結(jié)構(gòu)化綁定

  • constexpr if 表達(dá)式

  • map支持merge和extract

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“c++17新特性是什么”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識等著你來學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI