右值引用和移动语义

什么是左值和右值

左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边。定义时const修饰符后的左值,不能给他赋值,但是可以取它的地址。左值引用就是给左值的引用,给左值取别名。

右值也是一个表示数据的表达式,如:**字面常量、表达式返回值,函数返回值(这个不能是左值引用返回)**等等,右值可以出现在赋值符号的右边,但是不能出现出现在赋值符号的左边,右值不能取地址。右值引用就是对右值的引用,给右值取别名。

常量,表达式,函数返回值,直接引用因为权限放大会报错,不能引用临时变量和常量,加上const可以

使用右值引用不会报错

move

使用move可以把左值变成右值,但是只是返回值是右值,不会改变原来变量的属性。

万能引用,完美转发

函数模板使用右值引用,就是万能引用,你传什么就是什么引用

右值引用应用场景

c++11给类新增了,移动构造和移动赋值的成员函数。

比起普通的拷贝构造,通过重复利用了将亡值的资源,节省了构造时间。

对于以下类,如果我们没有移动构造,在接收函数返回的对象时就会浪费时间。

如果有移动构造,那么编译器检测到这个返回值是将亡值,会自动去掉移动构造,节省时间。

下面是这个类的代码

cpp 复制代码
class A
{
public:
	A(const char* s,int n)
	{
		cout << "A(const char* s,int n)" << endl;
		_s = new char[n];
		_n = n;
		for (int i = 0; i < n; i++)
			_s[i] = s[i];
	}
	A(const A& a)//拷贝构造
	{
		cout << "A(const A& a)" << endl;
		_s = new char[a._n];
		_n = a._n;
		for (int i = 0; i < _n; i++)
			_s[i] = a._s[i];
	}
	A( A&& a)//移动构造
	{
		cout << "A( A&& a)" << endl;
		swap(_s, a._s);
		swap(_n, a._n);
	}
	A& operator=(const A& a)//赋值运算符重载
	{
		cout << "A& operator=(const A& a)" << endl;
		_s = new char[a._n];
		_n = a._n;
		for (int i = 0; i < _n; i++)
			_s[i] = a._s[i];
		return *this;
	}
	A& operator=(A&& a)//移动赋值
	{
		cout << "A& operator=(A&& a)" << endl;
		swap(a._s, _s);
		swap(a._n, _n);
		return *this;
	}

	~A()
	{
		//cout << "~A()" << endl;
		delete[] _s;
	}
private:
	char* _s = nullptr;
	int _n = 0;
};
A fun()
{
	A tmp = A("333", sizeof("333"));
	return tmp;
}

int main()
{
	A a("444", sizeof("444"));
	a = fun();
}
相关推荐
clint4564 天前
C++进阶(1)——前景提要
c++
夜悊4 天前
C++代码示例:进制数简单生成工具
c++
郝学胜_神的一滴4 天前
CMake 021: IF 条件判据详诠
c++·cmake
_wyt0015 天前
洛谷 B3930 [GESP202312 五级] 烹饪问题 题解
c++·gesp
玖玥拾5 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
один but you5 天前
constexpr函数
c++
凡人叶枫5 天前
Effective C++ 条款41:了解隐式接口和编译期多态
java·开发语言·c++·effective c++
凡人叶枫5 天前
Effective C++ 条款42:了解 typename 的双重意义
java·linux·服务器·c++
小胖xiaopangss5 天前
BRpc使用
c++·rpc
-森屿安年-5 天前
63. 不同路径 II
c++·算法·动态规划