右值引用、引用折叠、移动语义、完美转发(言简意赅一文看懂)

348 阅读1分钟
右值引用(实现移动语义和完美转发,消除不必要的拷贝):

左值:等号左边,可以取地址
右值:等号右边,不可以取地址

左值引用&
右值引用&&
左值引用只能绑定左值,右值引用只能绑定右值
    
//将左值转化为右值,给需要右值的地方把左值转化为右值传递右值
std::move()
    

引用折叠规则
1.所有右值引用折叠到右值引用上仍然是一个右值引用。(A&& && 变成 A&&)
2.所有的其他引用类型之间的折叠都将变成左值引用。 (A& & 变成 A&; A& && 变成 A&; A&& & 变成 A&)
移动语义(传递的是临时对象的话,原来的东西拷贝完就没用了,新的还得再申请释放):
    
//拷贝构造函数
A(A& a)
//移动构造函数
A(A&& a)
//拷贝赋值函数
A& operator=(A& a)
//移动赋值函数
A& operator=(A&& a)
  • copy和move的区别
完美转发(一个函数给另一个函数传参时候,原参数是左值/右值,新函数还能保持左值/右值,就是完美转发):

//不完美的转发
void process(int& i){
    cout << "process(int&):" << i << endl;
}
void process(int&& i){
    cout << "process(int&&):" << i << endl;
}

void myforward(int&& i){
    cout << "myforward(int&&):" << i << endl;
    process(i);
}
int main(){
	myforward(2);  //本来2是右值;到了myforward里有了名字i,再传递process时候变成了传递左值
}


//用来实现完美转发,forward可以实现左右值的相互转换,move只能左值到右值
std::forward<T>(u)
原则:a.T为左值引用时,u被转换为左值;b.否则u被转换为右值
    
std::forward<int>(x) //x转换成右值,b
std::forward<int &>(x) //x转换成左值,a
std::forward<int &&>(x) //x转换成右值,b