C++ decltype 說明符
1.語法
decltype
( 實體 ) (1) (C++11 起)
decltype
( 表達式 ) (2) (C++11 起)
解釋:
1) 如果實參是沒有括號的標識表達式或沒有括號的類成員訪問表達式,那麼 decltype
產生以該表達式命名的實體的類型。如果沒有這種實體或該實參指名瞭一組重載函數,那麼程序非良構。
- 如果實參是指名某個結構化綁定的沒有括號的標識表達式,那麼
decltype
產生其被引用類型(在關於結構化綁定聲明的說明中有所描述)。 - 如果實參是指名某個非類型模板形參的沒有括號的標識表達式,那麼
decltype
生成該模板形參的類型(當該模板形參以占位符類型聲明時,類型會先進行任何所需的類型推導)。
2) 如果實參是其他類型為 T 的任何表達式,且
- 如果 表達式 的值類別是亡值,將會
decltype
產生 T&&; - 如果 表達式 的值類別是左值,將會
decltype
產生 T&; - 如果 表達式 的值類別是純右值,將會
decltype
產生 T。
如果 表達式 是返回類類型純右值的函數調用,或是右操作數為這種函數調用的逗號表達式,那麼不會對該純右值引入臨時量。
如果 表達式 是除瞭(可帶括號的)立即調用以外的 (C++20 起)純右值,那麼不會從該純右值實質化臨時對象:即這種純右值沒有結果對象。
該類型不需要是完整類型或擁有可用的析構函數,而且類型可以是抽象的。此規則不適用於其子表達式:decltype(f(g())) 中,g() 必須有完整類型,但 f() 不必。
註意:
- 如果對象的名字帶有括號,那麼它會被當做通常的左值表達式,從而 decltype(x) 和 decltype((x)) 通常是不同的類型。
- 在難以或不可能以標準寫法進行聲明的類型時,decltype 很有用,例如 lambda 相關類型或依賴於模板形參的類型。
2.關鍵詞decltype
示例:
#include <iostream> #include <type_traits> struct A { double x; }; const A* a; decltype(a->x) y; // y 的類型是 double(其聲明類型) decltype((a->x)) z = y; // z 的類型是 const double&(左值表達式) template<typename T, typename U> auto add(T t, U u) -> decltype(t + u) // 返回類型依賴於模板形參 { // C++14 開始可以推導返回類型 return t+u; } int main() { int i = 33; decltype(i) j = i * 2; std::cout << "i = " << i << ", " << "j = " << j << '\n'; std::cout << "i 和 j 的類型相同嗎?" << (std::is_same_v<decltype(i), decltype(j)> ? "相同" : "不同") << '\n'; auto f = [](int a, int b) -> int { return a * b; }; decltype(f) g = f; // lambda 的類型是獨有且無名的 i = f(2, 2); j = g(3, 3); std::cout << "i = " << i << ", " << "j = " << j << '\n'; }
輸出:
i 和 j 的類型相同嗎?相同
i = 33, j = 66
i = 4, j = 9
到此這篇關於C++ decltype
說明符的文章就介紹到這瞭,更多相關decltype 說明符內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 從c++標準庫指針萃取器談一下traits技法(推薦)
- C++ decltype用法舉例說明
- 深入淺析C++ traits技術
- C++右值引用與移動構造函數基礎與應用詳解
- 解析C++11的std::ref、std::cref源碼