我们平常说的引用,一般来说都会默认为左值引用。即为某一个对象起一别名,是间接访问的一种实现方式。
引用有许多的作用,例如传参中避免发生拷贝,在一定程度上我们可以提升性能。又例如传参时,实参为数组,我们知道数组是不能整体拷贝,故我们也可以通过引用来实现。
左值,右值
左值,右值都是表达式的属性。C++中的表达式要么是右值,要么是左值。而C++中某一表达式是左值还是右值?总的来说应该根据该表达式所起的作用来决定,如果该表达式的求值结果代表的是对象的身份(在内存中的位置),则该表达式是左值属性。如果该表达式的求值结果是一个值,则是右值属性。
那么,有左值引用,为什么又需要右值引用呢?
引入右值引用,是为了支持移动操作,避免拷贝。
例如我们现在实现一个自己设计一个类,其中成员函数包括push_back函数。
数据成员data为:
std::shared_ptr<std::vector<T>> data;
成员函数:
void push_back(const T& t) { data->push_back(t); }
如果此时我们调用push_back(520)
520为字面常量,左值引用不能绑定到字面常量,故在此时会产生一个临时对象来存储这个字面常量,之后左值引用才能绑定到该临时对象。
因此我们应当重载一个右值绑定类型的函数,从而能直接实现右值绑定到字面常量,避免不必要的开销,提高效率。
void push_back(T &&t) { data->push_back(std::move(t)); }
其中std::move是专用于将给定的左值转换为对应的右值引用类型。
要注意,如果调用了std::move函数后,除了对原对象赋值或者销毁它以外,已经不能再使用它了。
例如
std::string str="Hello";
a.push_back(std::move(str));
std::cout<<str;
此时str是一个空字符串了,已经不能再使用了。