左右值再探与decltype

Posted wuduojia

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了左右值再探与decltype相关的知识,希望对你有一定的参考价值。

Decltype 类型指示符

“引用从来都作为其所指对象的同义词出现,只有用在decltype处是一个例外

理解:

Decltype和auto区别:

1.     auto是从表达式类型推断出要定义的变量类型(这里的表达式是auto声明语句里的右值表达式),且用该表达式的值去初始化。而decltype虽然也是从表达式类型推断要定义的变量类型,但是不会用该表达式的值去初始化(这里的表达式是左值表达式?)

2.     decltype的结果类型与表达式形式密切相关(表达式只有变量和表达式不止有变量)

第二点就是理解这里的关键,对于decltype来说,如果表达式是变量名加上括号得到的就是一个引用(加括号的时候编译器将其视为表达式而不是变量):

int i=0;

decltype((i))d;  //d是一个int引用,必须初始化

decltype(i)d;    //d是一个普通的int

因此对于:

Decltype(*p)c    //c是一个int&,即引用。

在这里,引用和*p联系到了一起,这就是那个例外。

那么,为什么会是这样呢?

此外,为什么变量加个括号,在decltype中就会返回引用类型呢?

这一切都和区别2和左右值在decltype中的应用有关。

 

首先我们要明确,decltype中,表达式只有变量不会出现这种差异,变量是什么就返回什么;只有在表达式不止是变量时会有不同。

表达式之所以不同,是因为decltype此时是根据表达式的结果来返回对应的类型.那么表达式的结果有哪些呢?

1.表达式的结果对象能作为一条赋值语句的左值时,decltype返回的是引用类型

想到这,有必要再次深入了解C++中的左右值区别了。之前存的文章:

https://stackoverflow.com/questions/3601602/what-are-rvalues-lvalues-xvalues-glvalues-and-prvalues

以及最近在看的网络课笔记:

http://www.cnblogs.com/wuduojia/articles/7635635.html

派上了用场,接下来就是自己试着合起来理解一下。

 

首先是stack overflow里高票答案的理解:

一切的原因可能是因为移动时全语义的变化(全语义的意思见网络课笔记)。一旦我们对表达式进行移动而非复制,很容易就能看出在不同表达式上移动的区别。根据我对草案的理解,我认为rvalue和lvalue的概念和以前一样,只是在移动时更加细分了这个概念。

为什么需要它?也许并不是我们想的那样即定义不一样了,而是定义的更加精确了而已。

 

1.lvalue=left value=左值

2.xvalue =expiring value=消亡值

3.glvalue =generalized lvalue=泛左值

4.rvalue=right value=右值

5.prvalue =pure rvalue=纯右值

由于英语差,感觉自己理解的这五种值不对,又去查了资料:

https://www.zhihu.com/question/22111546

https://www.2cto.com/kf/201602/489942.html

https://www.zhihu.com/question/39846131

 

发现这些答案应该可以说清楚了,但是很多基础性的东西还不懂,无法理解!所以决定先放一放,以后再回头理解!

暂时先按照这篇文章的理解:

http://www.cnblogs.com/wuchanming/p/3751847.html

“使用关键字decltype的时候,左值和右值也有所不同。如果表达式的求值结果是左值,decltype作用于该表达式(不是变量)得到一个引用类型。举个例子,假定p的类型是int* ,因为解引用运算符生成左值,所以decltype(*p)的结果是int&。另一方面,因为解地址运算符生成右值,所以decltype(&p)的结果是int**,也就是说,结果是一个指向整型指针的指针;赋值也会产生引用,引用的类型就是左值的类型:

int a = 3, b = 4;

decltype(a = b)d = a;              //d是int&

 

即暂时理解为对象是左值,值是右值。当表达式返回的值是左值给decltype时得出的结果是引用。

所以:

int i=42;*p=&i;&r=i;

decltype (r+0)b;    //r+0的结果是一个具体值42,是一个右值,所以返回给decltype的类型是int

decltype(*p)c;       //*p是解引用p,得到的是一个对象,所以返回给decltype的类型是int引用

以上是关于左右值再探与decltype的主要内容,如果未能解决你的问题,请参考以下文章

再探jQuery

再探ELK-全新的ELK-5.2.0

老老实实学WCF] 第五篇 再探通信--ClientBase

再探共享存储

再探Java基础——throw与throws

网络嗅探与欺骗