右值引用
<c++ primer> 13.6节
怎么判断右值
左值持久,右值短暂
右值一般时字面常量,或者表达式求值中创建的临时结果(没人管的孩子),将很快被销毁。
获得右值引用-std::move
例如 auto f2 = std::move(foo_1) 来获得一个对象的右值引用
这并不会实际转移资源,但约定不应该再使用foo_1的值了。实际的资源转移应该在移动赋值和移动构造函数里面实现。
使用对象的移动构造函数和移动赋值函数
- 例如避免vector扩容时复制对象,定义move constructor,并显示指明 noexcept
std::vector<Foo> foo_array
for (size_t i = 0; i < 10; ++i)
{
// foo_array.push_back(Foo()); // 插入vector时,采用copy construct
foo_array.push_back(std::move(Foo())); // 插入vector时,采用move construct
}
- 使用右值引用,插入时就避免使用复制构造函数,临时的资源不被复制。
- 移动构造函数,注明为noexcept时,vector的动态扩容,也会采用移动构造函数,而不是复制构造
- 移动构造函数中要注意将源对象中的指针设置nullptr,避免析构时销毁原始的资源
- 默认的移动构造和移动赋值函数:
-
当一个类没有定义拷贝控制成员,并且任何非static数据成员都能移动操作时,编译器才会生成默认的移动成员