首頁  >  文章  >  後端開發  >  c++17新特性有哪些

c++17新特性有哪些

青灯夜游
青灯夜游原創
2021-01-29 11:57:1512805瀏覽

c 17新特性有:1、對auto表達式推導的規則進行了改變;2、lambda表達式可以捕捉「*this」;3、新增inline變量,可以直接將全域變數定義在頭檔中;4、條件式中支援初始化語句;5、枚舉的直接列表初始化等等。

c++17新特性有哪些

本教學操作環境:windows7系統、C 17版本、Dell G3電腦。

相關推薦:《C 影片教學

C 17中的新功能

1. auto關鍵字

從c 11開始,auto關鍵字能夠透過初始化器推導出變數的型別。在c 14中,auto關鍵字的能力進一步提升,能夠透過return語句推導出函數的回傳類型。使用auto關鍵字能夠提高編碼效率,同時也能簡化重構流程。但是,C 11中的auto推導,往往結果與預期的不同。

c 11 中為了支援統一初始化,引入了新的統一初始化語法,如下所示。

// 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>

這三種方式初始化的變量,最終型別推導的結果都是 std::initializer_list , 而不是我們認為的int。這是因為 當用於auto宣告變數的表達式是{}括起來的,推導的型別就會變成 std​​::initializer_list。

在C 17中,對auto表達式推導的規則進行了改變

// 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

對比發現, auto x5{3}, 會直接將變數推導成x5,而x3{1, 2} 這種方式也會編譯失敗。 auto推導的規則變得更直觀。

2. lambda表達式

lambda也是c 11中引入的,在C 11中,lambda表達式只能用捕捉this,this是當前物件的一個只讀的引用。在C 17中,可以擷取*this, *this是目前物件的拷貝,擷取目前物件的拷貝,能夠確保目前物件釋放後, lambda表達式能安全的呼叫this中的變數與方法。

3. inline變數

Inline 變數, inline變數可以讓變數有多於一次的定義。 C 17之前,我們定義全域變量, 總需要將變數定義在cpp檔中,然後在透過extern關鍵字來告訴編譯器 這個變數已經在其他地方定義過了。 inline變數出現後,我們可以直接將全域變數定義在頭檔中,而不用擔心出現redefine的錯誤訊息。

4. 條件表達式中支援初始化語句

c 17中支援在if 或switch 語句中進行初始化, 這個能力的出現能夠讓程式碼更加的簡潔。

// 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的標準函式庫也進行了擴充, 新增了以下幾種資料型別:

1. std::variant

std::variant是類型安全的聯合體,是一個加強版的union,variant支援更複雜的資料類型,例如map ,string等等

2. std::optional

std::optional表示一個可能存在的值。當我們透過函數建立一個物件時,通常使用透過函數傳回錯誤碼,而透過出參返回物件本身。如果透過optional傳回所建立的實例,就會變得更直觀,

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

has_value()   // 检查对象是否有值
value()       // 返回对象的值,值不存在时则抛出 std::bad_optional_access 异常
value_or()    // 值存在时返回值,不存在时返回默认值

3. std::any

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

4. std::string_view

string_view我最早使用的是boost版本的,c 17中的string_view 和boost類似。 string_view可以理解成原始字串一個唯讀引用。 string_view 本身沒有申請額外的記憶體來儲存原始字串的data, 僅僅保存了原始字串位址和長度等資訊。在很多情況下,我們只是臨時處理字串,本不需要對原始字串的一份拷貝。使用string_view可以減少不必要的記憶體拷貝,可以提高程式效能。相比使用字串指針,string_view做了更好的封裝。

要注意的是,string_view 由於沒有原始字串的所有權,使用string_view 一定要注意原始字串的生命週期。當原始的字串已經銷毀,則不能再呼叫string_view。

其他特性:

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

  • bool 表達式不能用, – 這兩個自增(減)運算子了

  • c 17中異常已經成為了型別系統的一部分,

  • 枚舉的直接列表初始化

  • #結構化綁定

  • constexpr if 表達式

  • map支援merge和extract

    #

更多程式相關知識,請造訪:程式設計學習! !

以上是c++17新特性有哪些的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn