咱们以前使用的typeid运算符来查询一个变量的类型,这种类型查询在运行时进行。RTTI机制为每个类型产生一个type_info类型的数据,而typeid查询返回的变量相应type_info数据,经过name成员函数返回类型的名称。同时在C++11中typeid还提供了hash_code这个成员函数,用于返回类型的惟一哈希值。RTTI会致使运行时效率下降,且在泛型编程中,咱们更须要的是编译时就要肯定类型,RTTI并没有法知足这样的要求。编译时类型推导的出现正是为了泛型编程,在非泛型编程中,咱们的类型都是肯定的,根本不须要再进行推导。程序员
而编译时类型推导,除了咱们说过的auto关键字,还有本文的decltype。编程
decltype与auto关键字同样,用于进行编译时类型推导,不过它与auto仍是有一些区别的。decltype的类型推导并非像auto同样是从变量声明的初始化表达式得到变量的类型,而是老是以一个普通表达式做为参数,返回该表达式的类型,并且decltype并不会对表达式进行求值。函数
int i = 4; decltype(i) a; //推导结果为int。a的类型为int。
using size_t = decltype(sizeof(0));//sizeof(a)的返回值为size_t类型 using ptrdiff_t = decltype((int*)0 - (int*)0); using nullptr_t = decltype(nullptr);
vector<int >vec; typedef decltype(vec.begin()) vectype; for (vectype i = vec.begin; i != vec.end(); i++) { //... }
这样和auto同样,也提升了代码的可读性。code
在C++中,咱们有时候会赶上一些匿名类型,如:ip
struct { int d ; doubel b; }anon_s;
而借助decltype,咱们能够从新使用这个匿名的结构体:字符串
decltype(anon_s) as ;//定义了一个上面匿名的结构体
这也是decltype最大的用途了。编译器
template <typename _Tx, typename _Ty> auto multiply(_Tx x, _Ty y)->decltype(_Tx*_Ty) { return x*y; }
标记符指的是除去关键字、字面量等编译器须要使用的标记以外的程序员本身定义的标记,而单个标记符对应的表达式即为标记符表达式。例如:hash
int arr[4]
则arr为一个标记符表达式,而arr[3]+0不是。编译
咱们来看下面这段代码:模板
int i=10; decltype(i) a; //a推导为int decltype((i))b=i;//b推导为int&,必须为其初始化,不然编译错误
仅仅为i加上了(),就致使类型推导结果的差别。这是由于,i是一个标记符表达式,根据推导规则1,类型被推导为int。而(i)为一个左值表达式,因此类型被推导为int&。
经过下面这段代码能够对推导四个规则做进一步了解
int i = 4; int arr[5] = { 0 }; int *ptr = arr; struct S{ double d; }s ; void Overloaded(int); void Overloaded(char);//重载的函数 int && RvalRef(); const bool Func(int); //规则一:推导为其类型 decltype (arr) var1; //int 标记符表达式 decltype (ptr) var2;//int * 标记符表达式 decltype(s.d) var3;//doubel 成员访问表达式 //decltype(Overloaded) var4;//重载函数。编译错误。 //规则二:将亡值。推导为类型的右值引用。 decltype (RvalRef()) var5 = 1; //规则三:左值,推导为类型的引用。 decltype ((i))var6 = i; //int& decltype (true ? i : i) var7 = i; //int& 条件表达式返回左值。 decltype (++i) var8 = i; //int& ++i返回i的左值。 decltype(arr[5]) var9 = i;//int&. []操做返回左值 decltype(*ptr)var10 = i;//int& *操做返回左值 decltype("hello")var11 = "hello"; //const char(&)[9] 字符串字面常量为左值,且为const左值。 //规则四:以上都不是,则推导为本类型 decltype(1) var12;//const int decltype(Func(1)) var13=true;//const bool decltype(i++) var14 = i;//int i++返回右值
这里须要提示的是,字符串字面值常量是个左值,且是const左值,而非字符串字面值常量则是个右值。
这么多规则,对于咱们写代码的来讲不免太难记了,特别是规则三。咱们能够利用C++11标准库中添加的模板类is_lvalue_reference来判断表达式是否为左值:
cout << is_lvalue_reference<decltype(++i)>::value << endl;
结果1表示为左值,结果为0为非右值。
一样的,也有is_rvalue_reference这样的模板类来判断decltype推断结果是否为右值。
参考资料:《深刻理解C++11》