右值引用和移动语义

什么是左值和右值

左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边。定义时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();
}
相关推荐
wanghu202444 分钟前
ABC460_E题题解
c++·算法
智者知已应修善业1 小时前
【51单片机象棋快棋赛 电子裁判器】2023-12-27
c++·经验分享·笔记·算法·51单片机
晚风予卿云月1 小时前
二分算法练习
数据结构·c++·算法·竞赛·算法随笔
lilili也2 小时前
C++:文件操作
c++
Lhan.zzZ2 小时前
C++多线程——std::thread与condition_variable形象理解
c++
头歌实践平台2 小时前
C++面向对象 - 运算符重载的应用
开发语言·c++·算法
思麟呀2 小时前
C++11并发编程:互斥锁
linux·开发语言·c++·windows
AC赳赳老秦2 小时前
OpenClaw批量任务队列优化:解决任务堆积、执行缓慢、优先级混乱问题
java·大数据·数据库·c++·自动化·php·openclaw
晚风予卿云月2 小时前
《二分答案》算法练习
数据结构·c++·算法·二分·竞赛·算法随笔
郭涤生2 小时前
C++ 各类数据的内存分区与读写性能详解
开发语言·c++