decltype 指定符

来自cppreference.com
< cpp‎ | language

检查实体的声明类型或表达式的类型及值分类。

目录

[编辑] 语法

decltype ( entity ) (1) (C++11 起)
decltype ( expression ) (2) (C++11 起)

[编辑] 解释

1) 若参数是无括号的 id 表达式并指名结构化绑定,则 decltype 产出被引用类型(描述于结构化绑定声明的规定)。
(C++17 起)
2) 若参数为无括号的 id 表达式或无括号的类成员访问表达式,则 decltype 产出以此表达式命名的实体的类型。若无这种实体,或该参数命名一些重载函数,则程序为病式。
3) 若参数是任何其他类型为 T 的表达式,且
a)expression值类别亡值,则 decltype 产出 T&&
b)expression的值类别为左值,则 decltype 产出 T&
c)expression 的值类别为纯右值,则 decltype 产出 T

expression 是一个返回类类型纯右值的函数调用,或一个右运算数为这种调用的逗号表达式,则不会对此纯右值引入 (C++17 前)实质化 (C++17 起)临时对象。类类型不需要完整或拥有可用的析构函数。此规则不应用于子表达式: decltype(f(g())) 中, g() 必须有完整类型,但 f() 不必。

注意若对象名称被括号括着,则它被当做通常的左值表达式,从而 decltype(x)decltype((x)) 通常是不同的类型。

decltype 在声明难以或不可能以标准记号声明的类型时适用,例如 lambda 相关类型或依赖于模板形参的类型。

[编辑] 关键词

decltype

[编辑] 示例

#include <iostream>
 
struct A { double x; };
const A* a = new A{0};
 
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) // 返回类型依赖于模板形参
{
    return t+u;
}
 
int main() 
{
    int i = 33;
    decltype(i) j = i * 2;
 
    std::cout << "i = " << i << ", "
              << "j = " << 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 = 33, j = 66
i = 4, j = 9

[编辑] 参阅

auto 指定符 指定表达式所定义的类型(C++11)[编辑]
(C++11)
在不求值语境中获取到其参数的引用
(函数模板) [编辑]