大家好,我是梁唐。
这是EasyC++系列的第38篇,来聊聊右值引用。
想要追求更好阅读体验的同学,可以访问github仓库:EasyLeetCode。
右值引用
左值和右值
在我们之前的文章当中,介绍的都是左值引用。C++11在左值引用的基础上推出了右值引用,由于是新特性,加上使用的频率也不是很高,有一定的学习成本。
我们先把引用这个概念抛开,先来看看什么是左值什么是右值。其实很简单,左值可以取地址,位于等于号的左侧。而右值没办法取地址,位于等于号的右侧。
1 | C++复制代码int a = 4; |
比如我们定义了一个int
型的变量a,让它的值等于4。其中a位于等于号的左侧,并且我们可以求a的地址。而4位于等于号的右侧,我们没有办法对4取地址。所以a是左值,4是右值。
再比如:
1 | C++复制代码int test() { |
同样,a位于等于号的左侧,有办法取地址是个左值。而test()
是一个临时值没办法取地址,是个右值。
所以到这里就比较清楚了,有地址的变量就是左值,没有地址的常量值、临时变量就是右值。
左值引用和右值引用
明白了左值、右值的概念再来看看左值引用、右值引用就简单了。左值引用顾名思义就是能够指向左值不能指向右值的引用。
1 | C++复制代码int a = 4; |
但是左值引用也有例外,就是使用const
修饰的左值引用是可以指向右值的:
1 | C++复制代码const int &b = 4; |
因为const
修饰的引用无法再更改,所以可以指向右值。如果大家度过STL代码的话,会发现其中一些函数的入参是const &
目的就是为了能够兼容传参是常量的情况。比如vector
当中的push_back
:
1 | C++复制代码void push_back (const value_type& val); |
右值引用和左值引用的概念类似,也就是能够指向右值但不能指向左值的引用。为了和左值引用区别, 使用&&
也就是两个&
符。老实讲这个符号很令人费解,因为它和and
的含义相同。
1 | C++复制代码int a = 4; |
上面第三行代码非法的原因是c是一个右值引用,它不能指向左值。如果我们非要指向呢?也不是没有办法,可以使用std::move
函数,它可以将一个左值转换成右值。
1 | C++复制代码using namespace std; |
move
函数听起来似乎是移动的意思,但其实它并没有移动变量,只不过做了一个类似于类型转换的操作。
不知道大家看到这里有没有觉得头大,其实还没有结束,还有一点很重要。即左值引用和右值引用这两者本身都是左值引用:
1 | C++复制代码void test(int && tmp) { |
C++中的引用是一个非常大的范畴,除了左值引用、右值引用之外还有非常多的细节。比如万能引用、引用折叠、完美转发等……
本文转载自: 掘金