右值引用和移动语义

什么是左值和右值

左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边。定义时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();
}
相关推荐
leaves falling9 分钟前
C/C++ 的内存管理,函数栈帧详讲
java·c语言·c++
wuyoula19 分钟前
AI导航智能决策系统源码 附教程
c++·tcp/ip·源码
浅念-40 分钟前
从LeetCode入门位运算:常见技巧与实战题目全解析
数据结构·数据库·c++·笔记·算法·leetcode·牛客
XY_墨莲伊2 小时前
【编译原理】实验二:基于有穷自动机FA词法分析器设计与实现
c语言·开发语言·c++·python
小辉同志2 小时前
74. 搜索二维矩阵
c++·leetcode·矩阵·二分查找
fpcc2 小时前
信号处理与AI中的卷积的关系
c++·人工智能·信号处理
t***5443 小时前
能否给出更多现代C++设计模式的示例
开发语言·c++·设计模式
梵尔纳多3 小时前
OpenGL 骨骼动画
c++·图形渲染·opengl
智者知已应修善业3 小时前
【51单片机独立按键控制往复流水灯启停】2023-6-13
c++·经验分享·笔记·算法·51单片机
t***5443 小时前
这些设计模式在现代C++中如何应用
java·c++·设计模式