检查实体的声明类型,或表达式的类型和值类别。

语法

decltype ( 实体 ) (1) (C++11 起)
decltype ( 表达式 ) (2) (C++11 起)

解释


1) 若实参是指名某个结构化绑定的无括号的标识表达式,则 decltype 产生被引用类型(在关于结构化绑定声明的说明中有所描述)。
(C++17 起)

2) 若实参是指名某个非类型模板形参的无括号的标识表达式,则 decltype 生成该模板形参的类型(当该模板形参以占位符类型声明时,则为进行任何所需的类型推导后的类型)。
(C++20 起)

3) 若实参为无括号的标识表达式或无括号的类成员访问表达式,则 decltype 产生以此表达式命名的实体的类型。若无这种实体,或该实参指名某个重载函数,则程序非良构。

4) 若实参是其他类型为 T 的任何表达式,且

a) 若 表达式 的值类别亡值,则 decltype 产生 T&&

b) 若 表达式 的值类别为左值,则 decltype 产生 T&

c) 若 表达式 的值类别为纯右值,则 decltype 产生 T


若 表达式 是返回类类型纯右值的函数调用,或是右操作数为这种函数调用的逗号表达式,则不对该纯右值引入临时量。
(C++17 前)

若 表达式 是除了(可带括号的)立即调用以外的 (C++20 起)纯右值,则不从该纯右值实质化临时对象:这种纯右值无结果对象。
(C++17 起)

不需要该类型完整或拥有可用的析构函数,而且类型可以是抽象的。此规则不适用于其子表达式:decltype(f(g())) 中,g() 必须有完整类型,但 f() 不必。

注意如果对象的名字带有括号,则它被当做通常的左值表达式,从而 decltype(x) 和 decltype((x)) 通常是不同的类型。

在难以或不可能以标准写法进行声明的类型时,decltype 很有用,例如 lambda 相关类型或依赖于模板形参的类型。

关键词

decltype

示例

运行此代码

  1. #include <iostream>
  2.  
  3. struct A { double x; };
  4. const A* a;
  5.  
  6. decltype(a->x) y; // y 的类型是 double(其声明类型)
  7. decltype((a->x)) z = y; // z 的类型是 const double&(左值表达式)
  8.  
  9. template<typename T, typename U>
  10. auto add(T t, U u) -> decltype(t + u) // 返回类型依赖于模板形参
  11. { // C++14 开始可以推导返回类型
  12. return t+u;
  13. }
  14.  
  15. int main()
  16. {
  17. int i = 33;
  18. decltype(i) j = i * 2;
  19.  
  20. std::cout << "i = " << i << ", "
  21. << "j = " << j << '\n';
  22.  
  23. auto f = [](int a, int b) -> int
  24. {
  25. return a * b;
  26. };
  27.  
  28. decltype(f) g = f; // lambda 的类型是独有且无名的
  29. i = f(2, 2);
  30. j = g(3, 3);
  31.  
  32. std::cout << "i = " << i << ", "
  33. << "j = " << j << '\n';
  34. }

输出:

  1. i = 33, j = 66
  2. i = 4, j = 9