十九、C++11
6. 右值引用和移动语句
右值引用使用场景和意义
在左值引用出来后,解决了大部分拷贝消耗。左值引用解决了形参传递的拷贝消耗,解决了部分返回对象拷贝的问题(如出了函数作用域,返回的对象还在,就可以使用左值引用减少拷贝)。但是 左值引用并没有解决返回对象是一个局部变量,出了函数作用域生命周期就到了,只能传值返回的拷贝 。
此时,右值引用的出现就解决了这个问题。右值引用借助移动语句,延长了资源的生命周期,类似于一种交换。他将原本要释放的资源交换到另一个对象里,避免了拷贝。
右值引用也只是解决了深拷贝的问题,深拷贝对象传引用返回只需要移动资源即可。浅拷贝的问题并没有解决,但是浅拷贝的代价比较小,也不需要解决。
移动构造和移动赋值
不知道大家有没有这样使用过:
#include <iostream>
#include <string>
#include <list>
using namespace std;
int main()
{
list<int> li;
li.push_back(666);
list<string> ls;
ls.push_back("123123");
return 0;
}
当我们往list中插入时,由于每插入一个数据,list就需要构造一个节点,数据赋值的时候就要产生拷贝,但是我们要是写一个右值引用+移动语义的构造函数,就可以将资源直接与右值交换,避免了拷贝。
ListNode(T&& x)
:_next(nullptr)
, _prev(nullptr)
, _data(move(x))
{}
这里没有调用深拷贝的拷贝构造,而是调用了移动构造,移动构造中没有新开空间,拷贝数据,所以效率提高了。
移动赋值也是:
string& operator=(string&& s)
{
cout << "string& operator=(string&& s) -- 移动赋值" << endl;
swap(s);
return *this;
}
STL库里的容器都是加上了移动赋值和移动构造。
等等等等。
右值引用引用左值
按照语法,右值引用只能引用右值,但右值引用一定不能引用左值吗?并不是,可以通过move函数来引用左值。move 函数并不搬移任何东西,唯一的功能就是将一个左值强制转化为右值引用,然后实现移动语义。
int main()
{
string s1("hello world");
// 这里s1是左值,调用的是拷贝构造
string s2(s1);
// 这里我们把s1 move处理以后, 会被当成右值,调用移动构造
// 但是这里要注意,一般是不要这样用的,因为我们会发现s1的
// 资源被转移给了s3,s1被置空了。
string s3(std::move(s1));
return 0;
}
STL容器的插入接口也增加了右值引用方法:
等等等。