decltype 是 c++11 用于编译时类型推导的关键字,根据表达式形式直接得出类型而不计算其值。其核心规则为:若表达式是标识符或类成员访问,返回声明类型;若是函数调用,返回函数返回类型;若为左值且不满足前两条,推导为 T&,否则为 T。例如 decltype(x) 得 int,decltype((x)) 因 (x) 为左值得 int&,decltype(3+4) 为右值得 int。在函数模板中常与尾置返回类型结合使用:auto func(T t, U u) -> decltype(t + u),使返回类型依赖参数表达式。与 auto 不同,decltype 保留引用和 const 属性,而 auto 去除顶层 cv 限定符。实际应用包括获取迭代器类型、定义表达式别名、SFINAE 条件编译及捕获 Lambda 类型等,如 decltype(vec.begin()) it = vec.end(); 提升泛型编程的灵活性与安全性。

decltype 是 C++11 引入的关键字,用于在编译时推导表达式的类型。它不计算表达式的值,而是根据表达式的形式直接得出其类型,常用于泛型编程、模板定义以及返回类型推导等场景。
基本语法与规则
decltype 的语法形式为:
decltype(expression)
其类型推导遵循以下三条核心规则:
- 如果表达式是标识符或类成员访问,decltype 返回该标识符的声明类型
- 如果表达式是函数调用或重载,decltype 返回函数返回值的声明类型
- 如果表达式是一个左值(即可以取地址的表达式),但不符合前两条,decltype 推导为 T&;否则推导为表达式的类型 T
例如:
int x = 5;
decltype(x) a = x; // a 的类型是 int
decltype((x)) b = x; // (x) 是左值表达式,b 的类型是 int&
decltype(3 + 4) c = 7; // 表达式结果是右值,c 的类型是 int
在函数返回类型中的应用
当函数模板的返回类型依赖于参数表达式时,使用 decltype 可以实现延迟返回类型声明。
C++11 提供了尾置返回类型(trailing return type)语法:
立即学习“C++免费学习笔记(深入)”;
template <typename T, typename U>
auto add(T t, U u) -> decltype(t + u) {
return t + u;
}
这里 auto 不是自动推导,而是与 -> 配合使用,真正的类型由 decltype(t + u) 决定。
与 auto 的区别
auto 和 decltype 都用于类型推导,但行为不同:
- auto 忽略引用和顶层 const,推导的是“初始化表达式的值类型”
- decltype 严格按表达式形式推导,保留引用和 const 属性
例如:
const int cx = 10;
const int& rx = cx;
auto ax = rx; // ax 的类型是 int(去除了 const 和 &)
decltype(rx) dr = cx; // dr 的类型是 const int&
实际应用场景
decltype 常用于以下情况:
- 编写通用容器遍历代码时获取迭代器类型
- 配合模板实现 SFINAE 技术进行条件编译
- 定义基于表达式的别名类型,如 typedef decltype(obj.func()) func_ret_t;
- 在 lambda 表达式中获取其类型(lambda 无名称,可用 decltype 获取)
比如:
std::vector<int> vec;
decltype(vec.begin()) it = vec.end(); // it 的类型是 vector<int>::iterator
基本上就这些。decltype 提供了精确的类型推导能力,尤其在复杂模板编程中非常有用,理解其规则有助于写出更安全、灵活的泛型代码。


