C++ Primer笔记(2.5.3)—— decltype类型指示符
#include<iostream>
/**********************************************************************/
/* 有时会遇到这种情况:希望从表达式的类型推断出要定义的变量类型,但是 */
/* 不想用该表达式的值初始化变量。 */
/* 说明符decltype,的作用是选择并返回操作数的数据类型。在此过程中, */
/* 编译器分析表达式并得到它的值,却不实际计算表达式的值。 */
/**********************************************************************/
int f()
{
int i = 1;
return i;
}
int main()
{
/* ----- 例子 ----- */
//编译器分析表达式并得到它的值,却不实际计算表达式的值。
decltype(f()) sum = 3; //sum的类型就是函数f的返回类型
std::cout << "sum的类型为:" << typeid(sum).name() << std::endl;
std::cout << std::endl;
/* 编译器并不实际调用函数f(),而且使用当调用发生是f()的返回值类型作为sum的类型 */
//decltype处理顶层const和引用的方式与auto有些不同。
//如果decltype使用的表达式是一个变量,则decltype返回该变量的类型(包括顶层const)
const int ci = 0, &cj = ci;
decltype(ci) x = 0; //x的类型是const int
decltype(cj) y = x; //y的类型是const int&,y绑定到变量x
//decltype(cj) z; //错误:z是一个引用,必须初始化
/* 引用从来都是作为其所指对象的同义词出现,只有在decltype中是个例外 */
/* ---------------- */
/* ----- decltype和引用 ----- */
//如果decltype使用的表达式不是一个变量,则decltype返回表达式结果对应的类型。
//有些表达式将向decltype返回一个引用类型。一般来说这种情况发生时,意味着该
//表达式的结果对象能作为一条赋值语句的左值:
//⚪decltype的结果可以是引用类型
int i = 42, *p = &i, &r = i;
decltype(r + 0) b; //正确:加法结果是int,那b也是int
//decltype(*p) c; //错误:c是int& 必须初始化
//⚪如果表达式的内容是解引用操作,则decltype将得到引用类型。
//解引用指针可以得到指针所指的对象,而且还能给这个对象赋值。
decltype(*p) pi = i;
/* 如果decltype使用的是一个不加括号的变量,则得到的结果就是该变量的类型; */
/* 如果给变量家上一层或多层括号,编译器就会把它当成是一个表达式 */
//decltype的表达式如果是加上了括号的变量,结果将是引用
decltype(i) e; //正确:e是一个(未初始化的)int
//decltype((i)) d; //错误:d是int&,必须初始化
/* 切记:decltype((variable))(注意是双层括号)的结果永远是引用 */
/* decltype(variable)结果只有当variable本身是引用时才是引用 */
/* -------------------------- */
return 0;
}