您好,登錄后才能下訂單哦!
這篇文章主要介紹了c++17新特性是什么,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
c++17新特性有:1、對auto表達式推導的規則進行了改變;2、lambda表達式可以捕獲“*this”;3、新增inline變量,可以直接將全局變量定義在頭文件中;4、條件表達式中支持初始化語句;5、枚舉的直接列表初始化等等。
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新特性是什么”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。