C++11之前,我们所说的引用只有左值引用(auto &a),现在我们通常使用的引用也是左值引用,但是准确一点说,C++11之后,除了左值引用(LVaule),还有右值引用(RValue)。
一句话概况LVaule和RValue
auto a = 10;
//其中 a就是左值, 10就是右值
auto b = a;
//其中 b是左值,a 是右值
C++11后,
L-value(左值):指可寻址的,L表示location,
A value that has an address.
R-value (右值):R表示read,
A value that does not have an address in a computer language.
左值: 如果一个表达式可以引用到某一个对象,并且这个对象是一块内存空间且可以被检查和存储,那么这个表达式就可以作为一个左值。
右值:引用了一个存储在某个内存地址里的“数据”。
Example
比如我们重写一个tuple的排序
vector<tupe<int, int, int>> v;
auto cmp = [](const auto &a, const auto &b) {
auto &&[x1, y1, d1] = a;
auto &&[x2, y2, d2] = b;
//这里就使用到了右值引用
return d1 < d2;
};
sort(v.begin(), v.end(), cmp);
tips:
通常右值引用与std::move()搭配使用
举个栗子
auto a = 10;
auto &&c = a;//error
auto &&d = std::move(a);
//std::move()实际上只是做了一个类型转换而已,返回一个右值,
使用右值引用和move操作可以提高效率,节约时间和空间成本
举个栗子
void swap(T &a, T &b)
{
T tmp(a); //复制一份
b = a; //调用赋值函数(assignment operator),
a = tmp; //调用赋值函数(assignment operator)
}
void swap(T &a, T &b)
{
T &&tmp = std::move(a);
a = std::move(b);
b = std::move(tmp);
}
上述两个swap,第一个swap通过拷贝赋值来进行交换,而第二直接对其所属权进行移动以提高效率