【C++】C++11——左右值|右值引用|移动语义|完美转发

news2025/1/11 8:03:44

文章目录

    • 一、左值与右值
      • 1.概念
      • 2.引用
      • 3.注意
    • 二、右值引用的意义
      • 1.左值引用意义
      • 2.右值引用和移动语义
      • 3.容器新增
    • 三、万能引用
    • 四、完美转发

一、左值与右值

1.概念

左值是什么?右值是什么?

左值是一个表示数据的表达式(如变量名解引用的指针

我们可以获取它的地址,可以对它赋值.(const修饰后的左值不能给它赋值)

注意:左值既可以出现在赋值符号的左边,也可以出现在赋值符号的右边

int main()
{
	//左值
	int a = 10;
	const int c = 20;
	a = c;
	int* p = new int(0);
	return 0;
}

右值也是一个表示数据的表达式,如:字面常量、表达式返回值、函数返回值(不能是左值引用返回)。

右值可以出现在赋值符号的右边,但是不能出现在赋值符号的左边,右值不能取地址。

int main()
{
	double x = 1.1, y = 2.2;
	//右值
	10;
	x + y;
	func(x + y);

	return 0;
}

2.引用

左值引用与右值引用

C++11中新增了右值引用的特性,为了区分,把C++11之前的引用称为左值引用。

无论是左值引用还是右值引用,本质都是在给对象取别名

  • 左值引用

左值引用就是对左值的引用,给左值取别名,通过&来声明

int main()
{
	//左值
	int* p = new int(0);
	int a = 1;
	const int b = 2;
	//左值引用
	int*& rp = p;
	int& rrp = *p;
	int& ra = a;
	const int& rb = b;
	return 0;
}
  • 右值引用

右值引用就是对右值的引用,给右值取别名,通过&&来声明

int main()
{
	//右值
	double x = 1.1, y = 2.2;
	10;
	x + y;
	func(x, y);
	//右值引用
	int&& r1 = 10;
	double&& r2 = x + y;
	double&& r3 = func(x, y);

	return 0;
}

3.注意

注意:

  • 左值引用右值问题

左值不能引用右值,这会导致权限放大,右值可读不可写,而左值可读可写

但是有const修饰左值引用就能保证被引用的数据不会被修改了,所以const左值引用可以引用右值

所以const左值引用既可以引用左值,也可以引用右值:

int main()
{
	int a = 10;
	int& ra1 = a;
	//int& ra2 = 10;//10是右值,不能被左值引用

	//const左值引用既可以引用左值,也可引用右值
	const int& ra4 = a;
	const int& ra3 = 10;
	return 0;
}
  • 右值引用左值问题

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

但是右值引用可以引用move以后的左值

move函数是C++11提供的一个函数,被move后的左值就能被赋值给右值引用

int main()
{
	//右值引用右值
	int&& r1 = 10;
	
	//右值引用move后的左值
	int a = 10;
	int&& r2 = move(a);
	return 0;
}

了解一下:

为什么要有const右值引用:我们知道右值引用不可改变,那const右值引用有什么作用:
右值不可以取地址,但是右值取别名后,会导致右值被存储到特定位置,且可以取到该位置的地址,也就是说:不能取字面量10的地址,但是rr1引用后,可以对rr1取地址,如果不想rr1被修改,那就可以用const int&&rr1去引用。右值不能取地址,引用之后变成左值了

int main()
{
	double x = 1.1, y = 2.2;
	//右值引用
	int&& rr1 = 10;
	const double&& rr2 = x + y;

	rr1++;//可以被修改
	//rr2++;//不可以被修改
    cout<<&rr1<<endl;
    cout<<&rr2<<endl;
	return 0;
}

所以const右值引用的意义在于:右值不可修改,不可取地址;右值引用之后开空间存储下来,对于引用而言是左值,可以取地址,可以改变,这是为了移动构造,去移动换取资源,具体移动构造可见后面。


二、右值引用的意义

1.左值引用意义

左值引用的意义

1.做函数参数:减少拷贝,提高效率,可做输出型参数2.做函数返回值:减少拷贝,提高效率。引用返回,可修改返回对象

但是左值引用并没有彻底的解决问题:

左值引用左返回值时,并不能避免函数返回对象时不必要的拷贝操作

如果函数的返回的是一个局部的对象,该对象出了函数作用域就被销毁了,这种情况下就不能用左值引用作为返回值了,只能以传值的方式返回(深拷贝),这是左值引用的缺陷。

//左值引用尚未解决的问题场景
template<class T>
T func3(const T& x)
{
	T ret;
	//...
	return ret;//返回局部对象,出作用域就会销毁
}
int main()
{
    func3(v1);
	return 0;
}

又比如to_string的模拟实现:

	string to_string(int value)
	{
		bool flag = true;
		if (value < 0)
		{
			flag = false;
			value = 0 - value;
		}
		hwc::string str;
		while (value > 0)
		{
			int x = value % 10;
			value /= 10;
			str += (x + '0');
		}
		if (flag == false)
		{
			str += '-';
		}
		std::reverse(str.begin(), str.end());
		return str;
	}

str是局部变量,除了作用域就会销毁:

hwc::string ret = hwc::to_string(-1234);

此时如果调用,就会调用string的拷贝构造函数

**所以C++11自然就会出手引出了右值引用:右值引用的意义之一就是为了补齐左值引用的这个短板:传值返回的拷贝问题。**其二对于插入一些插入右值数据,也可以减少拷贝。

2.右值引用和移动语义

C++11对右值进行了区分:纯右值与将亡值

  • 内置类型表达式的值 —— 纯右值
  • 自定义类型表达式的值—— 将亡值

移动构造:移动构造也是一个构造函数,该构造函数的参数是右值引用,移动构造实际就是把传入右值的资源转移过来,避免了深拷贝,所以称为移动构造,就是移动别人的资源来进行构造

//拷贝构造
string(const string& s)
			:_str(nullptr)
			, _size(0)
			, _capacity(0)
		{
			cout << "string(const string& s)->深拷贝" << endl;
			string tmp(s._str);
			swap(tmp);//this->swap(tmp)
		}

//移动构造
		string(string&& s)//右值引用
			:_str(nullptr)
            ,_size(0)
	        ,_capacity(0)
		{
			cout << "string(const string& s) -- 移动拷贝" << endl;
			swap(s);
		}
int main()
{
	hwc::string s1("hello world");
	hwc::string s2(s1);//拷贝构造
	
	//move之后变成右值。将亡值
	hwc::string s3(move(s1));//移动构造

	return 0;
}

image-20230310130459034

把s1移动到s3中去了,移动将亡值。

移动构造的意义:

没有移动构造之前,拷贝构造采用const左值引用来接收,所以无论是左值还是右值都会调用拷贝构造

有了移动构造之后,采用的是右值引用接收,如果传入右值,就会调用移动构造

string的拷贝构造是深拷贝,而移动构造是通过swap函数移动资源,所以调用移动构造的代价消耗更小

image-20230310145014217

这个时候成本大大降低,无需深拷贝,直接资源转移。

编译器优化问题,这是之前说过的,这里重新复习一下:

如果返回局部对象时,会先用这个局部对象拷贝构造出一个临时对象,然后再用这个临时对象来拷贝构造我们接收返回值的对象;

编译器会优化成:只需要一次拷贝构造

image-20230310151627730

ps:右值引用swap()的是将亡值,拷贝构造中不能直接swap,因为对象不是将亡值:

int main()
{
    hwc::string s1("12345");
	hwc::string s2(s1);
    return 0;
}

移动赋值

移动赋值就是一个赋值运算符重载函数,参数是右值引用类型,移动赋值就是将传入右值的资源转移过来,这样就避免了深拷贝,这也是移动赋值的由来。

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

			return *this;
		}

//移动赋值
		string& operator=(string&& s)
		{
			cout << "string& operator=(string&& s)->移动赋值拷贝" << endl;
			swap(s);
			return *this;
		}
int main()
{
	hwc::string ret;
	ret = hwc::to_string(-1234);
}

image-20230310154017654

移动赋值的意义:

没有移动赋值前,原先operator=采用的是const左值引用接收参数,所以无论赋值时传入左值还是右值都会调用原来的operator=。但是移动赋值采用了右值引用接收参数,所以如果赋值时传入的是右值,那么调用的就是移动赋值函数。而string原来的operator=是深拷贝,而移动赋值通过swap把资源进行转移,代价比原先的operator=代价小。

image-20230310161637523

to_string返回局部对象时,调用移动构造生成一个临时对象,然后在调用移动赋值将临时对象资源转移到接收返回值的对象上,这个过程调用了两个函数但却只是资源的移动,不需要进行深拷贝。(右值引用延长生命周期:资源延长了)

总结:右值引用和左值引用减少拷贝的原理不太一样。左值引用是取别名直接起作用,右值引用是间接起作用,实现移动构造和移动赋值,在拷贝的场景中如果是右值(将亡值),转移资源。

3.容器新增

C++11之后,STL中容器就增加了移动构造与移动赋值:

比如string新增移动构造

image-20230310222602906

比如string新增移动赋值

image-20230310222645337

另外,C++11为STL容器的插入接口也增加了右值引用:

image-20230310223205714

image-20230310223444844

我们来看一看区别:在hwc命名空间里list插入接口没有实现右值引用:

image-20230310222017539

在std里list插入接口实现了右值引用:

image-20230310222057404

string类提供了移动构造函数,并且容器的push_back接口提供了右值引用版本,如果push_back函数传入的参数string对象也是一个右值,那么push_back函数就可以通过string的移动构造函数来进行资源的转移,避免了深拷贝,提高效率。


三、万能引用

右值引用本身是左值。

模板中&&并不是右值引用,而是万能引用,既能接收左值也能接收右值,同时也能接收const左值、const右值:

//万能引用
template <typename T>
void PerfectForward(T&& t)
{

}
int main()
{
	int x = 1;

	PerfectForward(x);//左值
	PerfectForward(10);//右值
	PerfectForward(move(x));//右值

	const int y = 20;
	PerfectForward(y);//const左值
	PerfectForward(move(y));//const右值
	return 0;
}

万能引用会根据传入实参的类型进行推导,如果传入的实参是一个左值,那么这里的形参t就是左值引用,如果传入的实参是一个右值,那么这里的形参t就是右值引用,同时t是可以++的,而如果加上const左值、cosnt右值t就不可以++。举个例子:

void Func(int& x)
{
	cout << "左值引用" << endl;
}
void Func(const int& x)
{
	cout << "const 左值引用" << endl;
}
void Func(int&& x)
{
	cout << "右值引用" << endl;
}
void Func(const int&& x)
{
	cout << "const 右值引用" << endl;
}
template<class T>
void PerfectForward(T&& t)
{
	Func(t);
}
int main()
{
	int x = 1;

	PerfectForward(x);//左值
	PerfectForward(10);//右值,右值引用再传递时属性是左值
	PerfectForward(move(x));//右值

	const int y = 20;
	PerfectForward(y);//const左值
	PerfectForward(move(y));//const右值
	return 0;
}

PerfectForward传递的参数分别是左值、右值、右值、const左值、const右值,但是结果都是左值

image-20230310234450382

这是因为右值引用后会导致右值被存储到特定的位置,此时右值具有左值的属性,可以被取地址也可以被修改,所以PerfectForward函数调用Func函数会将t识别为左值。

而如果想要在传递参数的过程之中保持右值的属性,这就需要用到完美转发了。


四、完美转发

如果想要在参数传递的过程中保持其原有的属性,则需要在传参时调用forward函数:

void Func(int& x)
{
	cout << "左值引用" << endl;
}
void Func(const int& x)
{
	cout << "const 左值引用" << endl;
}
void Func(int&& x)
{
	cout << "右值引用" << endl;
}
void Func(const int&& x)
{
	cout << "const 右值引用" << endl;
}
template<class T>
void PerfectForward(T&& t)
{
	Func(std::forward<T>(t));
}
int main()
{
	int x = 1;

	PerfectForward(x);//左值
	PerfectForward(10);//右值,右值引用再传递时属性是左值
	PerfectForward(move(x));//右值

	const int y = 20;
	PerfectForward(y);//const左值
	PerfectForward(move(y));//const右值
	return 0;
}

image-20230310235852118

用完美转发给简化list提供右值引用的push_back与insert接口:

namespace hwc
{
	template <class T>
	struct list_node
	{
		list_node<T>* _next;
		list_node<T>* _prev;
		T _data;

		list_node(const T& x)
			:_next(nullptr)
			, _prev(nullptr)
			, _data(x)
		{
		}
        //右值引用
		list_node(T&& x)
			:_next(nullptr)
			, _prev(nullptr)
			, _data(forward<T>(x))//完美转发
		{
		}

	};
	template<class T, class Ref, class Ptr>
	struct __list_iterator
	{
		typedef list_node<T> node;
		typedef __list_iterator<T, Ref, Ptr> Self;
		node* _pnode;
		__list_iterator(node*p)
			:_pnode(p)
		{
		}
        //......
	};


	template<class T>
	class list
	{
		typedef list_node<T> node;
	public:
		typedef __list_iterator<T, T&, T*> iterator;
		typedef __list_iterator<T, const T&, const T*> const_iterator;
        
		iterator begin()
		{
			return iterator(_head->_next);
		}

		iterator end()
		{
			return iterator(_head);
		}
		void empty_initialize()
		{
			_head = new node(T());
			_head->_next = _head;
			_head->_prev = _head;

			_size = 0;

		}

		list()
		{
			empty_initialize();
		}

        //左值引用
		void push_back(const T& x)
		{
			insert(end(), x);
		}
        //右值引用
		void push_back(T&& x)
		{
			insert(end(), forward<T>(x));//完美转发
		}
        //左值引用
		iterator insert(iterator pos,const T& x)
		{
			node* newnode = new node(x);
			node* cur = pos._pnode;
			node* prev = cur->_prev;

			newnode->_prev = prev;
			prev->_next = newnode;
			newnode->_next = cur;
			cur->_prev = newnode;

			++_size;
			return iterator(newnode);
		}
        //右值引用
		iterator insert(iterator pos, T&& x)
		{
			node* newnode = new node(forward<T>(x));//完美转发
			node* cur = pos._pnode;
			node* prev = cur->_prev;

			newnode->_prev = prev;
			prev->_next = newnode;
			newnode->_next = cur;
			cur->_prev = newnode;

			++_size;
			return iterator(newnode);
		}
		private:
			node* _head;
			size_t _size;
	};
}
int main()
{
	hwc::list<hwc::string> lt;
	hwc::string s1("111111");
	lt.push_back(s1);//左值——深拷贝

	lt.push_back(hwc::string("222222"));

	lt.push_back("3333333");
	return 0;
}

image-20230311081837375


本篇结束…

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

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

相关文章

学校学生心理测评系统

青少年在线心理测评系统 这款系统&#xff0c;是和北大合作开发&#xff0c;并真实用于线上测评场景&#xff0c;该项目有完整后台&#xff0c;以及学生管理等模块。 我们欢迎以下形式合作&#xff1a; 单纯研究项目。合作对该测评平台进行升级。单纯使用。 请联系我们 silv…

MyBatis里面用了多少种设计模式?

在MyBatis的两万多行的框架源码中&#xff0c;使用了大量的设计模式对工程架构中的复杂场景进行解耦&#xff0c;这些设计模式的巧妙使用是整个框架的精华。经过整理&#xff0c;大概有以下设计模式&#xff0c;如图1所示。图101类型&#xff1a;创建型模式▊ 工厂模式SqlSessi…

英飞凌Tricore原理及应用介绍04_中断请求及仲裁过程

目录1.概述2. 中断请求及过程仲裁3. 中断传到CPU会被即时响应吗&#xff1f;1.概述 在Tricore架构中允许有多个中断源包括片上外设及外部中断世间产生的中断请求&#xff0c;以打断中断服务的提供者如CPU或DMA通道&#xff0c;那你知道在Tricore里中断请求在内核中的仲裁及处理…

【java基础】ArrayList源码解析

文章目录基本介绍构造器指定初始容量默认创建通过集合创建添加add扩容机制批量添加addAll添加指定位置add添加多个元素到指定位置addAll删除删除指定元素remove删除指定索引元素remove条件删除removeIf批量删除removeAll修改修改指定位置set替换所有满足要求元素replaceAll一些…

vscode环境配置(支持跳转,阅读linux kernel)

目录 1.卸载clangd插件 2.安装C插件 3. 搜索框内输入 “intell”&#xff0c;将 C_Cpp&#xff1a;Intelli Sense Engine 开关设置为 Default。 4.ubuntu安装global工具 5.vscode安装插件 6.验证是否生效 7.建立索引 1.卸载clangd插件 在插件管理中卸载clangd插件 2.安…

课设-机器学习课设-实现新闻分类

✅作者简介&#xff1a;CSDN内容合伙人、信息安全专业在校大学生&#x1f3c6; &#x1f525;系列专栏 &#xff1a;课设-机器学习 &#x1f4c3;新人博主 &#xff1a;欢迎点赞收藏关注&#xff0c;会回访&#xff01; &#x1f4ac;舞台再大&#xff0c;你不上台&#xff0c;…

linux下安装SonarQube

目录1. 准备安装环境2. 安装postgres数据库3. 安装SonarQube4. 使用SonarQube1. 准备安装环境 这里安装SonarQube的系统环境是Red Hat Enterprise Linux release 8.7 &#xff0c;然后将jdk的压缩包&#xff08;jdk-17.0.2_linux-x64_bin.tar.gz&#xff09;和sonarQube的压缩…

Web Components学习(2)-语法

一、Web Components 对 Vue 的影响 尤雨溪在创建 Vue 的时候大量参考了 Web Components 的语法&#xff0c;下面写个简单示例。 首先写个 Vue 组件 my-span.vue&#xff1a; <!-- my-span.vue --> <template><span>my-span</span> </template>…

Spring——spring整合JUnit

JUnit定义: Junit测试是程序员测试&#xff0c;即所谓 白盒测试 &#xff0c;因为程序员知道被测试的软件如何&#xff08;How&#xff09;完成功能和完成什么样&#xff08;What&#xff09;的功能。 Junit是一套框架&#xff0c;继承TestCase类&#xff0c;就可以用Junit进行…

基于Selenium+Python的web自动化测试框架(附框架源码+项目实战)

目录 一、什么是Selenium&#xff1f; 二、自动化测试框架 三、自动化框架的设计和实现 四、需要改进的模块 五、总结 总结感谢每一个认真阅读我文章的人&#xff01;&#xff01;&#xff01; 重点&#xff1a;配套学习资料和视频教学 一、什么是Selenium&#xff1f; …

SpringBoot bean 加载顺序如何查看(源码解读)

背景 SpringBoot bean 加载顺序如何查看&#xff0c;想看加载了哪些bean&#xff0c; 这些bean的加载顺序是什么&#xff1f; 实际加载顺序不受控制&#xff0c;但会有一些大的原则&#xff1a; 1、按照字母顺序加载&#xff08;同一文件夹下按照字母数序&#xff1b;不同文件…

界面开发(4)--- PyQt5实现打开图像及视频播放功能

PyQt5创建打开图像及播放视频页面 上篇文章主要介绍了如何实现登录界面的账号密码注册及登录功能&#xff0c;还简单介绍了有关数据库的连接方法。这篇文章我们介绍一下如何在设计的页面中打开本地的图像&#xff0c;以及实现视频播放功能。 实现打开图像功能 为了便于记录实…

CobaltStrike攻击payload(有效载荷)介绍

HTA文档Office宏payload生成器有效载荷生成器windows可执行程序windows可执行程序windows stageless生成所有有效载荷HTA文档该模块为HTML Application attack&#xff08;HTML应用攻击&#xff09;。简单来说&#xff0c;就是这个包生成一个运行有效负载的HTML应用程序该模块下…

TCP UPD详解

文章目录TCP UDP协议1. 概述2. 端口号 复用 分用3. TCP3.1 TCP首部格式3.2 建立连接-三次握手3.3 释放连接-四次挥手3.4 TCP流量控制3.5 TCP拥塞控制3.6 TCP可靠传输的实现3.7 TCP超时重传4. UDP5.TCP与UDP的区别TCP UDP协议 1. 概述 TCP、UDP协议是TCP/IP体系结构传输层中的…

Flink 定时加载数据源

一、简介 flink 自定义实时数据源使用流处理比较简单&#xff0c;比如 Kafka、MQ 等&#xff0c;如果使用 MySQL、redis 批处理也比较简单 如果需要定时加载数据作为 flink 数据源使用流处理&#xff0c;比如定时从 mysql 或者 redis 获取一批数据&#xff0c;传入 flink 做处…

三、HTTP协议之三

文章目录一、HTTPS协议概述二、 HTTPS使用成本三、从HTTP到HTTPS四、HTTP协议的瓶颈五、双工通信的websocket六、HTTP2.0一、HTTPS协议概述 二、 HTTPS使用成本 HTTPS对性能的影响 https之所有安全是因为使用TLS(SSL)来加密传输. 三、从HTTP到HTTPS 了解个大概 第一步&#x…

react:二、jsx语法规则

目录 1.传输数据的xml和json格式举例 2.jsx语法规则 3.js语句跟js表达式的区别 4.jsx的小练习 1.传输数据的xml和json格式举例 2.jsx语法规则 1.定义虚拟DOM时&#xff0c;不要写引号。 2.标签中混入JS表达式时要用{}。 3.样式的类名指定不要用class&#xff0c;要用cla…

第十一章 寡头垄断市场中的企业决策

寡头垄断市场的定义、条件 寡头垄断市场&#xff1a;少数几家企业控制了某一行业的市场&#xff0c;供给该行业生产的大部分产品 寡头垄断市场应具备的条件&#xff1a; 一个行业或市场中&#xff0c;只有少数几家企业企业之间存在着相互制约、相互依存的关系新企业进入行业比…

golang大杀器GMP模型

golang 大杀器——GMP模型 文章目录golang 大杀器——GMP模型1. 发展过程2. GMP模型设计思想2.1 GMP模型2.2 调度器的设计策略2.2.1 复用线程2.2.2 利用并行2.2.3 抢占策略2.2.4 全局G队列2.3 go func()经历了那些过程2.4 调度器的生命周期2.5 可视化的CMP编程2.5.1 trace方式2…

【LeetCode】33. 搜索旋转排序数组、1290. 二进制链表转整数

作者&#xff1a;小卢 专栏&#xff1a;《Leetcode》 喜欢的话&#xff1a;世间因为少年的挺身而出&#xff0c;而更加瑰丽。 ——《人民日报》 目录 33. 搜索旋转排序数组 1290. 二进制链表转整数 33. 搜索旋转排序数组 33. 搜索旋转排序…