• (3)左右值再探与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引用

    wuduojia
  • 相关阅读:
    javascript实现限制上传文件的大小
    js事件
    表格展开伸缩以及ztree异步加载
    二分法
    php操作beanstalkd
    ubuntu安装操作HttpSQS高速队列
    mysql中安全函数
    php操作httpsqs
    jquery操作表单
    ajax长轮询实例
  • 原文地址:https://www.cnblogs.com/wuduojia/p/7637057.html
Copyright © 2020-2023  润新知