右值引用和移动语句(C++11)

news2025/1/23 17:51:15

左值引用和右值引用

回顾引用

我们之前就了解到了左值引用,首先我们要了解引用在编译器底层其实就是指针。具体来说,当声明引用时,编译器会在底层生成一个指针来表示引用,但在代码编写和使用时,我们可以像使用变量类似取别名的方式一样来操作引用,而不需要显式地使用指针符号。这使得引用更为方便,且看起来更直观,同时也能保证所获得的引用总是有效的。那什么是左值,什么又是右值呢??

 

什么是左值与右值

左值是表示数据的表达式(如变量名或解引用的指针),也可能在赋值符号右边,具有地址的、可寻址的表达式才是左值。(typename & )

右值是一个表达式,如:字面常量、表达式返回值,函数返回值等等,且右值不能出现在赋值符号左边不能进行取地址才是右值。(typename &&)

 

 右值、左值引用的使用

//左值引用右值
int main()
{
	double x = 1.1, y = 2.2;
	double& r1 = x;
	//double& r2 = x + y;//此时的x1+x2是右值,必须加const修饰才能引用
	const double& r2 = x + y;

	return 0;
}

1. 左值引用只能引用左值,不能引用右值。
2. 但是const左值引用既可引用左值,也可引用右值。


 

//右值引用左值
int main()
{
	double x = 1.1, y = 2.2;
	double&& rr1 = x+y;
	//double&& r2 = x;//此时的x1是左值,所以要将属性转换再引用
	double&& r2 = move(x);

	return 0;
}

1. 右值引用只能右值,不能引用左值。
2. 但是右值引用可以move以后的左值。(此时改变的其实是返回值属性


移动构造和移动赋值

引用的好处就是不需要进行拷贝,所以对于自定义类型而言,引用的好处就显得更为重要。那么右值引用的好处也是同样......

下面就简单的以my_string的构造相关函数进行举例:

class my_string
{
public:
	//构造函数
	my_string(const char* tmp="")
		:_size(strlen(tmp))
		, _capacity(_size)
	{
		_s = new char[_capacity + 1];//还有一个\0
		strcpy(_s, tmp);
	}
	// 拷贝构造
	my_string(const my_string& s)
		:_s(nullptr)
	{
		cout << "my_string(const my_string& s) -- 深拷贝" << endl;
	}
	// 赋值重载
	my_string& operator=(const my_string& s)
	{
		cout << "my_string& operator=(my_string s) -- 深拷贝" << endl;
	}

private:
	char* _s;
	size_t _size;
	size_t _capacity;
};

以to_string函数举例深拷贝的消耗: 

my_string to_string(int num)
{
	my_string ret;
	//.........
    //对ret操作
	return ret;//ret返回值会拷贝构造临时对象
}
int main()
{
	my_string s;
	s = to_string(1234);//调用赋值重载

}

其实经过分析我们知道,to_string函数内部创建了一个对象,所以在函数调用结束的时候就会释放该ret对象的空间,所以当我们返回的时候实质上是将ret存进一个临时对象中作为返回值,所以此时会发生第一次的深拷贝,最后返回值再赋值给s对象,所以又发生了一次深拷贝。此时就相当于一共复制拷贝了两份空间,再算上原来的就有三份空间,所以此时如果该空间不是存的to_string,而是存的是其他的大量数据的话,那么此时的消耗是极其大的。

此时其实我们要了解C++11中将右值分为两类:1.纯右值(内置类型的右值)2.将亡值(自定义类型的右值) 

而此时我们的ret出了作用域就销毁,而且还会创建临时对象,所以我们的ret其实会被视作将亡值,所以此时就可以看作是一个右值(编译器进行的特殊处理),所以我们就可以不用再调用拷贝构造来创建临时对象了呀,反正该对象也要销毁,何不直接将临时对象指向该将亡值呢。同理再进行赋值的时候同样也可以啊(函数返回值就是典型的右值,临时对象)

 所以就引出了移动构造和移动拷贝:

    void swap(my_string& s)
	{
		std::swap(_s, s._s);
		std::swap(_size, s._size);
		std::swap(_capacity, s._capacity);
	}
	// 移动构造
	my_string(my_string&& s)//此时编译器会选择更匹配的而不是const my_string&
		:_s(nullptr)
		, _size(0)
		, _capacity(0)
	{
		swap(s);
	}
	// 移动赋值
	my_string& operator=(my_string&& s)
	{
		swap(s);
		return *this;
	}

但是我们要注意的点就是,当我们实现了移动拷贝和移动赋值以后,那么我们将对象move修饰之后进行拷贝和赋值以后,会进行swap进行资源的转移,所以会将该move对象的资源交换,所以在进行移动拷贝之后,该move对象的资源交换后就没了,而调用移动赋值之后,move对象的资源就是交换了。

所以最好不要随意地对一个左值进行move修饰,可能会有意想不到的结果发生。 


其实C++11之后,对于容器的push函数都增加了移动构造的形式,这种尤其是对那些需要进行深拷贝的对象而言会方便很多。此时就不需要再拷贝一份,直接将资源交换即可完成拷贝工作。

 右值引用后的属性

  右值被右值引用之后的属性是左值所以就可以被修改:

int main()
{
    double x = 1.1, y = 2.2;
    int&& rr1 = 10;
    const double&& rr2 = x + y;
    rr1 = 20;//右值引用后可以修改
    rr2 = 5.5; // 报错

    return 0;
}

实例: 

 就该移动构造而言,s对象就是右值引用的过后的,再调用swap函数,此时在string&& s接受参数的时候其实s对象的属性就变成了左值,即可被修改,所以传给swap函数时用string& s接受就没问题,但是两个过程中自始至终依旧是一个对象,没有创建额外的对象。

 

就拿这上述string的push_back函数来说,我们知道C++11之后,容器的push_back函数都重载了一份右值插入。所以对于以上的函数来说就显得很麻烦了,因为我们知道右值引用接收右值以后,数据的属性就会变成左值,因此对于这种函数嵌套调用的情况就会显得十分繁琐,需要将每次参数都move一下才可以实现接下来的右值引用接收。但是我们move之后就相当于是写固定了,所以相较于每一次的move对象,我们用完美转发会更好。

 

完美转发 


void Fun(int &x){ cout << "左值引用" << endl; }
void Fun(const int &x){ cout << "const 左值引用" << endl; }
void Fun(int &&x){ cout << "右值引用" << endl; }
void Fun(const int &&x){ cout << "const 右值引用" << endl; }

// 模板中的&&不代表右值引用,而是万能引用,其既能接收左值又能接收右值。
// 模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的能力,
// 但是引用类型的唯一作用就是限制了接收的类型,后续使用中都退化成了左值,
// 我们希望能够在传递过程中保持它的左值或者右值的属性, 就需要用到完美转发
template<typename T>
void PerfectForward(T&& t)//万能引用
{
    Fun(t);//调用fun函数时并不清楚t对象的类型
}

std::forward 完美转发在传参的过程中保留对象原生类型属性

func( forward<T>(t) )//完美转发,保持原有属性

 新的类功能

其实我们的移动构造和移动赋值函数也会默认生成,但是条件是:类中没有实现析构函数 、拷贝构造、拷贝赋值重载(就是都没有实现才满足条件)对于默认生成的移动构造函数,对于内置类
型成员会执行逐成员按字节拷贝,自定义类型成员,则需要看这个成员是否实现移动构造,如果实现了就调用移动构造,没有实现就调用拷贝构造。
 

强制生成默认函数的关键字default
禁止生成默认函数的关键字delete

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

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

相关文章

鸿蒙绘制折线图基金走势图

鉴于鸿蒙下一代剥离aosp&#xff0c;对于小公司而言&#xff0c;要么用h5重构&#xff0c;要么等大厂完善工具、等华为出转换工具后跟进&#xff0c;用鸿蒙重新开发一套代码对于一般公司而言成本会大幅增加。但对于广大开发者来说&#xff0c;暂且不论未来鸿蒙发展如何&#xf…

中国消费电子行业发展趋势及消费者需求洞察|徐礼昭

一、引言 近年来&#xff0c;随着科技的飞速发展&#xff0c;消费电子行业面临着前所未有的挑战与机遇。本文将从行业发展趋势、消费者需求洞察以及企业数字化转型的方向和动作三个方面&#xff0c;对消费电子行业进行深入剖析。 二、消费电子行业发展趋势 5G技术的普及和应…

WEB安全之Python

WEB安全之python python-pyc反编译 python类似java一样&#xff0c;存在编译过程&#xff0c;先将源码文件*.py编译成 *.pyc文件&#xff0c;然后通过python解释器执行 生成pyc文件 创建一个py文件随便输入几句代码(1.py) 通过python交互终端 >>>import py_compil…

【C++练级之路】【Lv.1】C++,启动!(命名空间,缺省参数,函数重载,引用,内联函数,auto,范围for,nullptr)

目录 引言入门须知一、命名空间1.1 作用域限定符1.2 命名空间的意义1.3 命名空间的定义1.4 命名空间的使用 二、C输入&输出2.1 cout输出2.2 cin输入2.3 std命名空间的使用惯例 三、缺省参数3.1 缺省参数概念3.2 缺省参数分类 四、函数重载4.1 函数重载概念4.2 函数重载分类…

JavaSE自定义验证码图片生成器

设计项目的时候打算在原有的功能上补充验证码功能&#xff0c;在实现了邮箱验证码之后想着顺便把一个简单的图片验证码生成器也实现一下&#xff0c;用作分享。 注意&#xff0c;实际开发中验证码往往采用各种组件&#xff0c;通过导入依赖来在后端开发时使用相关功能&#xf…

泊车功能专题介绍 ———— 汽车全景影像监测系统性能要求及试验方法(国标未公布)

文章目录 术语和定义一般要求功能要求故障指示 性能要求响应时间图像时延单视图视野范围平面拼接视图视野平面拼接效果总体要求行列畸变拼接错位及拼接无效区域 试验方法环境条件仪器和设备车辆条件系统响应时间试验图像时延试验单视图视野范围试验平面拼接视图视野试验平面拼接…

【大学英语视听说上】Mid-term Test 2

Section A 【短篇新闻1】 You probably think college students are experts at sleeping, but parties, preparations for tests, personal problems and general stress can rack a students sleep habits, which can be bad for the body and the mind. Texas Tech Univer…

51爱心流水灯32灯炫酷代码

源代码摘自远眺883的文章&#xff0c;大佬是30个灯的&#xff0c;感兴趣的铁汁们可以去看看哦~&#xff08;已取得原作者的许可&#xff09;&#xff1a;基于STC89C51单片机设计的心形流水灯软件代码部分_单片机流水灯代码_远眺883的博客-CSDN博客 由于博主是个小菜鸡&#xff…

【Python从入门到进阶】43.验证码识别工具结合requests的使用

接上篇《42、使用requests的Cookie登录古诗文网站》 上一篇我们介绍了如何利用requests的Cookie登录古诗文网。本篇我们来学习如何使用验证码识别工具进行登录验证的自动识别。 一、图片验证码识别过程及手段 上一篇我们通过requests的session方法&#xff0c;带着原网页登录…

前缀和 LeetCode1423. 可获得的最大点数

几张卡牌 排成一行&#xff0c;每张卡牌都有一个对应的点数。点数由整数数组 cardPoints 给出。 每次行动&#xff0c;你可以从行的开头或者末尾拿一张卡牌&#xff0c;最终你必须正好拿 k 张卡牌。 你的点数就是你拿到手中的所有卡牌的点数之和。 给你一个整数数组 cardPoi…

一个网站,四种创建制作电子期刊的方法

想象一下&#xff0c;你正在走进一家神奇的商店&#xff0c;里面陈列着各种精美的杂志和期刊。但是&#xff0c;这些杂志和期刊并不是印刷品&#xff0c;而是可以直接在网站上制作和发布的电子期刊。 但是像这样能在网上发的电子期刊该怎么制作呢&#xff1f;不知道如何制作的小…

数字媒体技术基础之:常见字体类型

字体 Font在数字设计和排版中起着至关重要的作用&#xff0c;不同的字体类型为文本呈现和创意表达提供了丰富多样的可能性。 .fon 字体 .fon 文件是 Windows 早期系统中使用的一种字体文件格式。 特点&#xff1a; 1、基于像素的位图字体。 2、不支持无损缩放&#xff0c;主要用…

国际语音通知是什么?国际语音通知系统有哪些功能?

一、国际语音通知是什么&#xff1f; 如同国际短信通知&#xff0c;国际语音通知也在多种生活场景中扮演着重要的角色&#xff0c;如会议通知、商品发货通知、物流更新通知、快递取件通知、外卖取餐通知等。那么什么是语音通知呢&#xff1f; 国际语音通知可将通知的文本信息使…

算法复习,数据结构 ,算法特性,冒泡法动态演示,复杂度,辗转相除法*,寻找最大公因数

算法复习 知识点 1. 程序 数据结构 算法 2. 算法&#xff1a; 求解问题的策略数据结构&#xff1a;问题的数学模型程序&#xff1a;微计算机处理问题编制的一组指令 3. **特性 ** 有穷性&#xff1a;算法在执行有穷步后能结束确定性&#xff1a;每一指令有确切的含义&a…

规则引擎专题---5、Groovy环境搭建和基础语法

概述 Groovy是用于Java虚拟机的一种敏捷的动态语言&#xff0c;它是一种成熟的面向对象编程语言&#xff0c;既可以用于面向对象编程&#xff0c;又可以用作纯粹的脚本语言。使用该种语言不必编写过多的代码&#xff0c;同时又具有闭包和动态语言中的其他特性。 Groovy是从Jav…

线程安全的问题以及解决方案

线程安全 线程安全的定义 线程安全:某个代码无论是在单线程上运行还是在多线程上运行,都不会产生bug. 线程不安全:单线程上运行正常,多线程上运行会产生bug. 观察线程不安全 看看下面的代码: public class ThreadTest1 {public static int count 0;public static void main…

<JavaEE> synchronized关键字和锁机制 -- 锁的特点、锁的使用、锁竞争和死锁、死锁的解决方法

目录 一、synchronized 关键字简介 二、synchronized 的特点 -- 互斥 三、synchronized 的特点 -- 可重入 四、synchronized 的使用示例 4.1 修饰代码块 - 锁任意实例 4.2 修饰代码块 - 锁当前实例 4.3 修饰普通方法 - 锁方法所在实例 4.4 修饰代码块 - 锁指定类对象 …

threadlocal - 黑马程序员

目录 1、ThreadLocal介绍1.2 ThreadLocal基本使用1.2.1、常用方法1.2.2 使用案例 1.3 ThreadLocal类与synchronized关键字 2、运用场景_事务案例3、ThreadLocal的内部结构4、 ThreadLocal的核心方法源码5、ThreadLocalMap源码分析5.2 弱引用和内存泄漏 课程地址&#xff1a; ht…

【大学英语视听说上】“智力”口语问答练习

题目&#xff1a; book 2, page 9, question 4 回答&#xff1a; 1: What do you think of the view “Intelligence must be bred, not trained”? I think this view is biased. The view suggests that intelligence is primarily determined by genetic factors and inh…

drawio画图工具的四种使用方式

1、免安装使用&#xff08;绿色版&#xff09; 这种直接下载下来直接就可以使用&#xff0c;属于绿色版&#xff08;开箱即用&#xff09;&#xff0c;适用于个人 点击下载地址 2、 安装使用 这种下载下来就需要安装才可使用&#xff0c;适用于个人 点击下载地址 3、war包…