decltype那些事

1.基本使用

decltype的语法是:

  1. decltype (expression)

这里的括号是必不可少的,decltype的作用是“查询表达式的类型”,因此,上面语句的效果是,返回 expression 表达式的类型。注意,decltype 仅仅“查询”表达式的类型,并不会对表达式进行“求值”。

1.1 推导出表达式类型

  1. int i = 4;
  2. decltype(i) a; //推导结果为int。a的类型为int。

1.2 与using/typedef合用,用于定义类型。

  1. using size_t = decltype(sizeof(0));//sizeof(a)的返回值为size_t类型
  2. using ptrdiff_t = decltype((int*)0 - (int*)0);
  3. using nullptr_t = decltype(nullptr);
  4. vector<int >vec;
  5. typedef decltype(vec.begin()) vectype;
  6. for (vectype i = vec.begin; i != vec.end(); i++)
  7. {
  8. //...
  9. }

这样和auto一样,也提高了代码的可读性。

1.3 重用匿名类型

在C++中,我们有时候会遇上一些匿名类型,如:

  1. struct
  2. {
  3. int d ;
  4. doubel b;
  5. }anon_s;

而借助decltype,我们可以重新使用这个匿名的结构体:

  1. decltype(anon_s) as ;//定义了一个上面匿名的结构体

1.4 泛型编程中结合auto,用于追踪函数的返回值类型

这也是decltype最大的用途了。

  1. template <typename T>
  2. auto multiply(T x, T y)->decltype(x*y)
  3. {
  4. return x*y;
  5. }

完整代码见:decltype.cpp

2.判别规则

对于decltype(e)而言,其判别结果受以下条件的影响:

如果e是一个没有带括号的标记符表达式或者类成员访问表达式,那么的decltype(e)就是e所命名的实体的类型。此外,如果e是一个被重载的函数,则会导致编译错误。否则 ,假设e的类型是T,如果e是一个将亡值,那么decltype(e)为T&&否则,假设e的类型是T,如果e是一个左值,那么decltype(e)为T&。否则,假设e的类型是T,则decltype(e)为T。

标记符指的是除去关键字、字面量等编译器需要使用的标记之外的程序员自己定义的标记,而单个标记符对应的表达式即为标记符表达式。例如:

  1. int arr[4]

则arr为一个标记符表达式,而arr[3]+0不是。

举例如下:

  1. int i = 4;
  2. int arr[5] = { 0 };
  3. int *ptr = arr;
  4. struct S{ double d; }s ;
  5. void Overloaded(int);
  6. void Overloaded(char);//重载的函数
  7. int && RvalRef();
  8. const bool Func(int);
  9. //规则一:推导为其类型
  10. decltype (arr) var1; //int 标记符表达式
  11. decltype (ptr) var2;//int * 标记符表达式
  12. decltype(s.d) var3;//doubel 成员访问表达式
  13. //decltype(Overloaded) var4;//重载函数。编译错误。
  14. //规则二:将亡值。推导为类型的右值引用。
  15. decltype (RvalRef()) var5 = 1;
  16. //规则三:左值,推导为类型的引用。
  17. decltype ((i))var6 = i; //int&
  18. decltype (true ? i : i) var7 = i; //int& 条件表达式返回左值。
  19. decltype (++i) var8 = i; //int& ++i返回i的左值。
  20. decltype(arr[5]) var9 = i;//int&. []操作返回左值
  21. decltype(*ptr)var10 = i;//int& *操作返回左值
  22. decltype("hello")var11 = "hello"; //const char(&)[9] 字符串字面常量为左值,且为const左值。
  23. //规则四:以上都不是,则推导为本类型
  24. decltype(1) var12;//const int
  25. decltype(Func(1)) var13=true;//const bool
  26. decltype(i++) var14 = i;//int i++返回右值

学习参考:https://www.cnblogs.com/QG-whz/p/4952980.html