c 17新特性有:1、對auto表達式推導的規則進行了改變;2、lambda表達式可以捕捉「*this」;3、新增inline變量,可以直接將全域變數定義在頭檔中;4、條件式中支援初始化語句;5、枚舉的直接列表初始化等等。
本教學操作環境: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中文網其他相關文章!