文章目录
- 左值引用
- 左值引用的概念
- 左值引用的使用
- 右值引用
- 右值引用的概念
- 右值引用的使用
- 左右值相互引用
- 左值引用对右值进行引用
- 右值引用对左值进行引用
- 右值引用使用场景和意义
- 左值引用的优势
- 左值引用的短板
- 右值引用的优势
- 完美转发
- 模板万能引用
- 完美转发实际运用场景
左值引用
左值引用的概念
左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边
。定义时const修饰符后的左值,不能给他赋值,但是可以取它的地址。左值引用就是给左值的引用,给左值取别名。
左值引用的使用
右值引用
右值引用的概念
右值也是一个表示数据的表达式,如:字面常量、表达式返回值,函数返回值(这个不能是左值引用返回)等等
,右值可以出现在赋值符号的右边,但是不能出现出现在赋值符号的左边,右值不能取地址。右值引用就是对右值的引用,给右值取别名。
右值引用的使用
这时不知道大家有没有一个疑惑,右值引用后的别名可以修改右值吗?答案是可以的。
需要注意的是右值是不能取地址的,但是给右值取别名后,会导致右值被存储到特定位置,且可以取到该位置的地址
,也就是说例如:不能取字面量10的地址,但是rr1引用后,可以对rr1取地址,也可以修改rr1。如果不想rr1被修改,可以用const int&& rr1 去引用
,是不是感觉很神奇,这个了解一下实际中右值引用的使用场景并不在于此,这个特性也不重要。
左右值相互引用
左值引用对右值进行引用
我们知道左值引用可以引用左值,那么左值引用可以引用右值吗
?答案是可以的,不过要加const。
右值引用对左值进行引用
我们知道右值引用可以引用右值,那么右值引用可以引用左值吗
?答案是可以的,不过要加move。move以后的值具有右值属性。
那么右值引用对左值进行引用后,可以修改吗?
答案是可以的。
右值引用使用场景和意义
前面我们可以看到左值引用既可以引用左值和又可以引用右值
,那为什么C++11还要提出右值引用呢?是不是化蛇添足呢?下面我们来看看左值引用的短板,右值引用是如何补齐这个短板的!
左值引用的优势
场景一:
场景二:
左值引用的短板
我们知道左值引用都是对存在且还没销毁的值进行引用,但是如果一个值不存在或者销毁了,我们还能用左值引用吗
?答案是不能。
场景一:
我们先把右值引用的移动构造屏蔽掉,把左值引用的拷贝构造放开。
那么如果我们把右值引用的移动构造放开呢?
场景二:
==右值引用的拷贝构造放开: ==
我们说了这么多把左值引用的场景换成右值引用的场景,那么右值引用到底有什么用呢
?
右值引用的优势
右值分为纯右值和将亡值。
我们知道左值引用对左值资源的拷贝是没错的。但是左值引用对右值的将亡值拷贝
就有问题了,因为将亡值在使用一次后就会销毁掉。这时候就运用到了移动构造
。
移动构造本质是将参数右值的资源窃取过来,占为已有,那么就不用做深拷贝了,所以它叫做移动构造,就是窃取别人的资源来构造自己。
从这里我们可以看出,右值引用时是对资源的转移不是拷贝,所以右值引用的效率高。
完美转发
std::forward<T>(t)(完美转发)在传参的过程中保持了t的原生类型属性。
模板万能引用
从运行结果可以的得知模板的&&确实是万能,既能引用左值也能引用右值。
但是引用后就退化了,全部都变成了左值。那么该怎么解决呢?如何在传参的过程中保持自己的属性呢?这时候就运用到了完美转发。
完美转发实际运用场景
没有完美转发时:
有完美转发时: