C++右值引用/移动语义

news2025/1/23 9:15:14

在此之前,我们所用的引用,其实都是左值引用。

int a = 10;
int& ra = a;

下面我们来重新认识一下引用:

而何为左值?左值引用其实是什么?请往下看~

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

例如:

int main()
{
	//左值:变量名/解引用的指针
	// 以下的p、b、c、*p都是左值
	int* p = new int(0);  //p是变量名,为左值  *p为解引用的指针
	int b = 1;  // b为变量名
	const int c = 2;  //c为变量名。加上const后c的值不能被修改
	c = 3;//error

	//左值引用:
	// 以下几个是对上面左值的左值引用
	int*& rp = p;  //对p进行引用,rp为p的别名
	int& rb = b;
	const int& rc = c;
	int& pvalue = *p;  
	return 0;
}

右值是什么?右值引用呢?

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

右值引用是用两个&&表示,左值引用是用一个&表示。

int main()
{
	//右值:字面常量、表达式返回值、函数返回值

	double x = 1.1, y = 2.2;
	// 以下几个都是常见的右值
	10; //字面常量
	x + y;  //表达式返回值,即return x + y;
	fmin(x, y); //函数返回值

	// 以下几个都是对右值的右值引用
	int&& rr1 = 10;
	rr1 = 20;//注意这里!
	int* prr1 = &rr1; //注意这里!
	double&& rr2 = x + y;
	double&& rr3 = fmin(x, y);

	// 这里编译会报错:error C2106: “=”: 左操作数必须为左值
	10 = 1;
	x + y = 1;
	fmin(x, y) = 1;
	return 0;
}

从上面的代码我们看到两个神奇的地方!

第一个是在右值引用后,竟然可以对右值引用重新赋值!

第二个是右值是不能取地址的,但是给右值取别名后,这个别名可以被取地址!

这是右值引用的特性!右值是不能取地址的,但是给右值取别名后,会导致右值被存储到特定位置,且可以取到该位置的地址,也就是说例如:不能取字面量10的地址,但是rr1引用后,可以对rr1取地址,也可以修改rr1。如果不想rr1被修改,可以用const int&& rr1 去引用!

对于右值,我们可以将其分成两种右值:

第一种是纯右值。所谓纯右值,是内置类型表达式的值。

第二种是将亡值。所谓将亡值,是自定义类型表达式的值。

总结右值引用和左值引用:

基于上述对左值和右值的解析,我们可以总结以下几点:

①左值引用只能引用左值,不能引用右值。这里涉及到了权限的放大。

②如果加了const修饰的左值引用,可以引用左值和右值。因为有了const修饰,要么是权限的平移,要么是权限的缩小。

③右值引用只能引用右值,不能引用左值。

④右值可以引用move后的左值。函数move的作用是让左值变成右值去使用。

int main()
{
	// ①左值引用只能引用左值,不能引用右值。
	int a = 10; 
	int& ra1 = a; // ra1为a的别名,左值引用引用左值,a为左值
	//int& ra2 = 10; // 编译失败,因为10是右值,左值引用不能引用右值
	
	// ②const左值引用既可引用左值,也可引用右值。
	const int& ra3 = 10;
	const int& ra4 = a;


	// ③右值引用只能右值,不能引用左值。

	int&& r1 = 10;
	// error C2440: “初始化”: 无法从“int”转换为“int &&”
	// message : 无法将左值绑定到右值引用
	int a = 10;
	int&& r2 = a; //a是左值,&&是右值引用,右值引用不能引用左值
	// 右值引用可以引用move以后的左值
	int&& r3 = std::move(a);

	return 0;
}

右值引用的作用和意义

左值引用很好,因为左值引用的存在,在函数传参和函数传返回值的时候减少了拷贝,从而提高了效率。但是存在这样一个问题:在一个函数中,如果要使用左值引用作为返回值,那么这个返回的变量的生命周期必须不能随函数的销毁而销毁!

看下面代码:

//函数传参使用左值引用,减少拷贝
template<class T>
void func1(const T& x)
{

}

//函数传返回值,使用左值引用减少拷贝,
//返回的x是传进来的参数,x不会随函数的销毁而销毁。
template<class T>
const T& func2(const T& x)
{
	// ...

	return x;
}

//ret为函数栈帧中的变量,会随函数的销毁而销毁
//ret做返回值,这种情况下就得进行拷贝
template<class T>
T func3(const T& x)
{
	T ret;
	// ...
	return ret;
}

对于前两种情况,都可以使用左值引用,减少拷贝。但是对于第三种情况,在没有认识到右值引用的时候,它就必须进行拷贝。

这就是左值引用未能解决的场景问题。因此,右值引用的价值之一,就是补齐左值引用的这一个短板。

这里选择复用我们自己模拟实现string类的代码作为测试代码:模拟实现string类

这里将对string类添加右值引用的移动构造和移动赋值!

在此之前,我们先来看看,在没有实现移动构造和移动赋值之前,以下代码的结果如何:

string to_string(int value)
	{
		bool flag = true;
		if (value < 0)
		{
			flag = false;
			value = 0 - value;
		}

		my_string::string str;
		while (value > 0)
		{
			int x = value % 10;
			value /= 10;

			str += ('0' + x);
		}

		if (flag == false)
		{
			str += '-';
		}

		std::reverse(str.begin(), str.end());
		return str;
	}
int main()
{
	my_string::string ret;  
	ret = my_string::to_string(-1234);

	return 0;
}

结果如下: 在调用to_string的时候,传值传参,调用一次深拷贝。返回值返回的时候,因为没有移动构造,调用就会调用拷贝构造,因为const左值引用可以引用右值,因此会调用深拷贝。赋值给ret的时候也调用了一次深拷贝。

在代码中,-1234是常量,为右值。但是在模拟的string类中,并没有专门右值引用的构造,并且对ret的赋值,也是右值,但是没有右值引用的赋值,只能去调用深拷贝,因为深拷贝的左值引用加了const修饰。

接着我们加入右值引用的移动拷贝和移动构造:

在实现这两个接口前,在上文中提到了右值引用的两种形式:纯右值和将亡值。

我们来看将亡值,从名字中可以看出,这种右值是即将失去生命的值,也就是生命周期快到了,往后的代码程序中不需要它,因此,我们可以利用将亡值进行一次移动构造和移动赋值!

既然都将亡了,那么我就用我的吸星大法,将你的功力吸走!因此,实现的思路就是交换!

		// 移动构造
		string(string&& s)
		{
			cout << "string(const string& s) -- 移动拷贝" << endl;

			swap(s);
		}

		// 移动赋值
		string& operator=(string&& s)
		{
			cout << "string& operator=(string s) -- 移动赋值" << endl;
			swap(s);

			return *this;
		}

上述测试代码的结果变成了这样的!

我们自己实现的to_string,在传入-1234作为参数,此时拷贝的是使用移动拷贝,而str作为函数里面的变量,在返回值返回的时候,是作为右值调用移动赋值赋值给ret,减少了拷贝!

整体测试代码如下:

namespace my_string
{
	class string
	{
	public:
		typedef char* iterator;
		iterator begin()
		{
			return _str;
		}

		iterator end()
		{
			return _str + _size;
		}

		string(const char* str = "")
			:_size(strlen(str))
			, _capacity(_size)
		{
			//cout << "string(char* str)" << endl;

			_str = new char[_capacity + 1];
			strcpy(_str, str);
		}

		// s1.swap(s2)
		void swap(string& s)
		{
			::swap(_str, s._str);
			::swap(_size, s._size);
			::swap(_capacity, s._capacity);
		}

		// 拷贝构造
		string(const string& s)
		{
			cout << "string(const string& s) -- 深拷贝" << endl;

			string tmp(s._str);
			swap(tmp);
		}

		// 赋值重载
		string& operator=(const string& s)
		{
			cout << "string& operator=(string s) -- 深拷贝" << endl;
			string tmp(s);
			swap(tmp);

			return *this;
		}

		// 移动构造
		string(string&& s)
		{
			cout << "string(const string& s) -- 移动拷贝" << endl;

			swap(s);
		}

		// 移动赋值
		string& operator=(string&& s)
		{
			cout << "string& operator=(string s) -- 移动赋值" << endl;
			swap(s);

			return *this;
		}


		~string()
		{
			delete[] _str;
			_str = nullptr;
		}

		char& operator[](size_t pos)
		{
			assert(pos < _size);
			return _str[pos];
		}

		void reserve(size_t n)
		{
			if (n > _capacity)
			{
				char* tmp = new char[n + 1];
				strcpy(tmp, _str);
				delete[] _str;
				_str = tmp;

				_capacity = n;
			}
		}

		void push_back(char ch)
		{
			if (_size >= _capacity)
			{
				size_t newcapacity = _capacity == 0 ? 4 : _capacity * 2;
				reserve(newcapacity);
			}

			_str[_size] = ch;
			++_size;
			_str[_size] = '\0';
		}

		//string operator+=(char ch)
		string& operator+=(char ch)
		{
			push_back(ch);
			return *this;
		}

		const char* c_str() const
		{
			return _str;
		}
	private:
		char* _str = nullptr;
		size_t _size = 0;
		size_t _capacity = 0; // 不包含最后做标识的\0
	};

	//转化成字符串
	string to_string(int value)
	{
		bool flag = true;
		if (value < 0)
		{
			flag = false;
			value = 0 - value;
		}

		my_string::string str;
		while (value > 0)
		{
			int x = value % 10;
			value /= 10;

			str += ('0' + x);
		}

		if (flag == false)
		{
			str += '-';
		}

		std::reverse(str.begin(), str.end());
		return str;
	}
}


int main()
{
	my_string::string ret;  
	ret = my_string::to_string(-1234);


	return 0;
}

总结一下右值引用和左值引用对减少拷贝的方式:

右值引用和左值引用减少拷贝的原理不一样。左值引用减少拷贝的方法是起别名,直接起作用,而右值引用减少拷贝的方法是实现移动拷贝和移动赋值,在将亡值的情况下,直接转移资源,间接起作用。

右值引用的价值之二:对于插入一些插入右值数据,也可以减少拷贝!

比如list容器,如果插入接口insert的传值是左值引用,那么在插入右值的时候,由于没有移动拷贝,并且加了const修饰,此时的左值引用可以引用右值,调用的是深拷贝。

 

而如果使用了移动拷贝,参数为右值引用,那么就会采用移动拷贝,减少了拷贝的次数!

 未完待续......

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/397098.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

77. writerows写入多行

文章目录1. 目标任务2. 准备工作3. writerow单行写入4. writerows多行写入5. a以追加的模式写入值6. 总结1. 目标任务 新建【各班级成绩】文件夹&#xff1b; 在该文件夹下新建一个【1班成绩单.csv】文件&#xff1b; 在该文件中写入下面的内容&#xff1a; 成绩 姓名 刘一…

CentOS 8搭建EMQX集群

概览 EMQX (opens new window)是一款大规模可弹性伸缩的云原生分布式物联网 MQTT (opens new window)消息服务器。 EMQ X 设计目标是实现高可靠&#xff0c;并支持承载海量物联网终端的MQTT连接&#xff0c;支持在海量物联网设备间低延时消息路由: 1. 稳定承载大规模的 MQTT 客…

Allegro如何添加菜单栏操作指导

Allegro如何添加菜单栏操作指导 用Allegro设计PCB的时候,将常用的命令放在菜单栏的话可以方便使用,省去设计时间,菜单如下图 Allegro支持自由添加或者删除菜单,具体操作如下 点击View点击Customize Toolbar

【使用vue init和vue create的区别以及搭建vue项目的教程】

vue init 是vue-cli2.x的初始化方式&#xff0c;可以使用github上面的一些模板来初始化项目 webpack是官方推荐的标准模板名 使用方式&#xff1a;vue init webpack 项目名称 例如使用github上面electron-vue的模板使用方式&#xff1a;vue init electron-vue 项目名称教程目…

Java的数据库编程:JDBC

Content &#x1f389;1什么是API &#x1f389;2.什么是JDBC &#x1f389;3.数据库驱动包的安装 &#x1f389;4.数据库安装包在idea的使用 &#x1f389;5.JDBC的增删改查的简单实现 今天为大家带来JAVA的数据库编程,也就是用Java实现数据库 数据库的最基本的操作就是…

分布式锁简介

Redis因为单进程、性能高常被用于分布式锁&#xff1b;锁在程序中作用是同步工具&#xff0c;保证共享资源在同一时刻只能被一个线程访问。 Java中经常用的锁synchronized、Lock&#xff0c;但是Java的锁智能保证单机的时候有效&#xff0c;分布式集群环境就无能为力了&#xf…

软件设计师错题集

软件设计师错题集一、计算机组成与体系结构1.1 浮点数1.2 Flynn分类法1.3 指令流水线1.4 层次化存储体系1.4.1 程序的局限性1.5 Cache1.6 输入输出技术1.7 总线系统1.8 CRC循环冗余校验码二、数据结构与算法基础2.1 队列与栈2.2 树与二叉树的特殊性2.3 最优二叉树&#xff08;哈…

VisualSP Enterprise - February crack

VisualSP Enterprise - February crack VisualSP(可视化支持平台)提供了一个上下文中完全可定制的培训平台&#xff0c;它可以作为企业web应用程序的覆盖层提供。无论员工正在使用什么应用程序&#xff0c;他们都能够快速访问页面培训和指导&#xff0c;说明如何最有效地使用该…

C++基础了解-14-C++ 字符串

C 字符串 一、C 风格字符串 C 风格的字符串起源于 C 语言&#xff0c;并在 C 中继续得到支持。字符串实际上是使用 null 字符 \0 终止的一维字符数组。因此&#xff0c;一个以 null 结尾的字符串&#xff0c;包含了组成字符串的字符。 下面的声明和初始化创建了一个 RUNOOB …

教你如何搭建店铺—收支管理系统,demo可分享

1、简介1.1、案例简介本文将介绍&#xff0c;如何搭建店铺-收支管理。1.2、应用场景以店铺收支管理为核心&#xff0c;维度数据分析&#xff0c;智能指导门店经营&#xff0c;账目清晰一目了然&#xff0c;店铺经营更高效。2、设置方法2.1、表单搭建1&#xff09;新建表单【客户…

如何使用码匠连接 DynamoDB

目录 在码匠中集成 DynamoDB 在码匠中使用 DynamoDB 关于码匠 DynamoDB 是亚马逊 AWS 的一种高性能、全托管的 NoSQL 数据库服务。作为一种数据源&#xff0c;DynamoDB 能够提供高度可扩展性、低延迟和可靠性。它支持多种数据类型和数据模型&#xff0c;包括键-值、文档和图…

元宇宙时代来临,Facebook豪掷百亿是谋略还是赌博?

"Facebook向元宇宙发起冲击&#xff0c;豪掷百亿是谋略还是赌博&#xff1f;"2022年&#xff0c;Facebook宣布将投资100亿美元用于元宇宙技术的开发和推广。这笔巨额资金的投入是否会给Facebook带来巨大的回报&#xff0c;还是一场高风险的赌博呢&#xff1f;首先&am…

【13种css选择器】学css选择器,这一篇就够了

举例形象让你学会&#xff0c;不搞官方话css所有的选择器相邻兄弟选择器后续兄弟选择器后代选择器子代选择器并集选择器(多重选择器)属性选择器伪类选择器伪元素选择器class选择器&#xff08;类选择器&#xff09;id选择器*选择器&#xff08;通配符选择器&#xff09;标签选择…

【算法之旅】初识数据结构与算法

一名软件工程专业学生的算法之旅&#xff0c;记录自己从零开始学习数据结构与算法&#xff0c;从小白的视角学习数据结构&#xff1a;数组、对象/结构、字符串、队列、栈、树、图、堆、平衡树/线段树等&#xff0c;学习算法&#xff1a;枚举、排序、搜索、计数、分治策略、动态…

Windows中配置docker没有hyper-v功能解决方案

&#x1f468; 作者简介&#xff1a;大家好&#xff0c;我是Taro&#xff0c;前端领域创作者 ✒️ 个人主页&#xff1a;唐璜Taro &#x1f680; 支持我&#xff1a;点赞&#x1f44d;&#x1f4dd; 评论 ⭐️收藏 文章目录前言解决步骤&#xff1a;1.新建文档2. 另存为3. 功能…

Tomcat独立部署-Nginx-1.12.2配置SSL

目录 &#x1f3c6;1. 实现思路 &#x1f3c6;2. 重启服务器 &#x1f3c6;3. proxy_pass 后地址带/和不带/的url地址显示 &#x1f3c6;4. 配置SSL证书 &#x1f3c6;5. 遇到问题 &#x1f3c6;6. 参考文章 学习完本篇博客您将掌握&#xff1a; 1、使用Tomcat配置SSL域名…

SpringMVC中的拦截器不生效的问题解决以及衍生出的WebMvcConfigurationSupport继承问题思考

文章目录SpringMVC中的拦截器不生效的问题解决WebMvcConfigurationSupport继承问题思考SpringMVC中的拦截器不生效的问题解决 过滤器代码(被Spring扫描并管理)&#xff1a; Component public class StuInterceptor implements HandlerInterceptor {Overridepublic boolean pr…

Dynamics365安装失败解决及注册编写

一、修改错误昨天登录报错今天开始返回我之前设置的断点开始重新配置&#xff0c;Reporing Services配置完成后发现dynamics365还是下载失败之后下载了一上午dynamics365就一直卡在最后的界面进度条不动索性我直接把所有环境都卸载了 连同虚拟机卸载重装终于在下午的时候dynami…

设计模式---工厂模式

目录 1. 简单工厂模式 2. 工厂方法模式 1. 简单工厂模式 简单工厂模式(Simple Factory Patterm)又称为静态工厂方法模式(Static Factory Model)&#xff0c;它属于类创建型模式。在简单工厂模式中&#xff0c;可以根据参数的不同返回不同类的实例。简单工厂模式专门定义了一…

【Nginx】Nginx的安装配置

环境说明系统&#xff1a;Centos 7一、编译安装Nginx官网下载地址nginx: download#安装依赖 [rootnginx nginx-1.22.1]# yum install gcc pcre pcre-devel zlib zlib-devel -y #从官网下载Nginx安装包&#xff0c;并进行解压、编译、安装 [rootnginx ~]# wget https://nginx.or…