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

左值引用和右值引用

回顾引用

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

 

什么是左值与右值

左值是表示数据的表达式(如变量名或解引用的指针),也可能在赋值符号右边,具有地址的、可寻址的表达式才是左值。(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];//还有一个\0strcpy(_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.mzph.cn/news/193083.shtml

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

相关文章

HarmonyOS系统和Android系统有什么区别?

鸿蒙系统和安卓系统有如下几点区别&#xff1a;点击这里查看获取鸿蒙系统资料方式 (qq.com) 一、开发商不同&#xff1a; 鸿蒙OS&#xff1a;由中国华为公司主导开发的系统&#xff0c;2019年首次发布&#xff0c;现在已经更新至鸿蒙OS4.0。 安卓系统&#xff1a;是由安迪鲁宾…

visual Studio MFC 平台实现图像增强中的线性变换(负变换)和非线性变换(对数与幂律)

MFC 实现数字图像处理中的图像增强操作 本文使用visual Studio MFC 平台实现图像增强中典型的三种图像增强的方法中的两大类&#xff0c;包括线性变换–>负变换&#xff0c;非线性变换–>对数变换和幂律变换&#xff1b;其中第三大类分段式变换可以参考MFC实现图像增强–…

Android Termux 安装Kali Linux 或 kali Nethunter史诗级详细教程

Android Termux 安装Kali Linux 或 kali Nethunter史诗级详细教程 一、Termux配置1、下载安装2、配置存储和换源3、基本工具安装 二、Kali Linux安装1、下载安装脚本2、更换apt源3、图形化安装 三、Kali Nethunter安装1、下载安装脚本2、更换apt源3、图形化连接 四、报错汇总1、…

2023年5月电子学会青少年软件编程 Python编程等级考试一级真题解析(判断题)

2023年5月Python编程等级考试一级真题解析 判断题(共10题,每题2分,共20分) 26、在编写较长的Python程序时,所有代码都不需要缩进,Python会自动识别代码之间的关系 答案:错 考点分析:考查python代码书写格式规范,python编写较长的程序时,需要明确严格的缩进,不然有…

【ArcGIS Pro微课1000例】0044:深度学习--面部模糊(马赛克)

本文讲解ArcGIS Pro中通过深度学习工具实现人脸面部模糊,起到马赛克的作用。 文章目录 一、效果对比二、工具介绍三、案例实现一、效果对比 原始图片: 深度学习后的模糊照片: 二、工具介绍 本工具为ArcGIS Pro工具箱中的深度学习工具中的:使用深度学习分类像素,如下所示…

vue3中自定义hook函数

使用Vue3的组合API封装的可复用的功能函数 自定义hook的作用类似于vue2中的mixin技术 自定义Hook的优势: 很清楚复用功能代码的来源, 更清楚易懂 案例: 收集用户鼠标点击的页面坐标 hooks/useMousePosition.ts文件代码&#xff1a; import { ref, onMounted, onUnmounted …

Java LeetCode篇-深入了解关于栈的经典解法(栈实现:中缀表达式转后缀)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 中缀表达式转后缀说明 1.1 实现中缀表达式转后缀思路 2.0 逆波兰表达式求值 2.1 实现逆波兰表达式求值思路 3.0 有效的括号 3.1 实现有效的括号思路 4.0 栈的压…

法学毕业生个人简历16篇

想要从众多法学毕业求职者中脱颖而出&#xff0c;找到心仪的相关工作&#xff1f;可以参考这16篇精选的法学专业应聘简历案例&#xff0c;无论是应届比预算还是有工作经验&#xff0c;都能从中汲取灵感&#xff0c;提升简历质量。希望对大家有所帮助。 法学毕业生简历模板下载…

RPG项目01_脚本代码

基于“RPG项目01_场景及人物动画管理器”&#xff0c;我们创建一个XML文档 在资源文件夹下创建一个文件夹&#xff0c; 命名为Xml 将Xnl文档拖拽至文件夹中&#xff0c; 再在文件夹的Manager下新建脚本LoadManager 写代码&#xff1a; using System.Collections; using System…

Pycharm调用Conda虚拟环境

参考这个链接的评论区回答&#xff1a;Pycharm调用Conda虚拟环境 笑死&#xff0c;我之前也是这样的&#xff0c;不过好像也能用&#xff0c;搞不懂~

Ontrack EasyRecovery2024数据恢复软件详细功能介绍

Ontrack EasyRecovery2024是一款功能强大的数据恢复软件&#xff0c;它可以帮助用户从各种存储设备中恢复丢失或删除的数据。它支持多种文件系统和文件类型&#xff0c;可以恢复包括照片、视频、音频、文档、电子邮件和归档文件等不同类型的数据。 EasyRecovery15Mac版本下载如…

文案二次创作软件,文案二次创作的软件

文案创作成为品牌传播和营销不可或缺的一环。对于许多从业者而言&#xff0c;文案创作常常是一项既耗时又耗力的工作。为了解决这一文案创作的难题&#xff0c;市场上涌现出了众多的智能文案生成工具。我们通过对这些工具的介绍和分析&#xff0c;希望能够为你提供一些在文案创…

Micropython for QNX编译过程

Micropython for QNX编译过程 执行步骤 1. https://github.com/micropython/micropython select tag 1.20.0 git clone micropython 2. make -C mpy-cross 3. 修改py/mkenv.mk CROSS_COMPILE ntoaarch64- 注意如果这步必须在make -C mpy-cross 之后执行&#xff0c;如果需要重…

宝塔+docker+jenkins部署vue项目----笔记版

宝塔dockerjenkins部署vue项目&#xff08;保姆级教程&#xff09;https://blog.csdn.net/weixin_47284756/article/details/129339940 基于上述教程&#xff0c;不同的地方。 1.我使用的是gitee&#xff0c;所以需要在jenkins中安装gitee插件 配置gitee&#xff0c;其他默认配…

SmartsoftHelp8,条形码,二维码 生成,解析 专业工具

生成条形码 生成二维码 条形码解析 二维码解析 专业工具 下载地址&#xff1a; https://pan.baidu.com/s/1zBgeYsqWnSlNgiKPR2lUYg?pwd8888

大学程序员的养生之道

呀哈喽&#xff0c;我是结衣。 今天给大家带来的是大学程序员的养生之道&#xff01; 作为一名大学生还没有深刻的感受到未来的恐怖&#xff0c;但每当我看到这些对程序员的评价还是不禁感慨。 不要让自己的学习之路变成这样啊&#xff01;程序员的职业发展&#xff1a;某编程语…

CSS:calc() 函数 / 动态计算长度值 / 不同场景使用

一、理解 css calc() 函数 CSS calc() 函数是一个用于计算 CSS 属性值的函数。它可以在 CSS 属性值中使用数学表达式&#xff0c;从而实现动态计算属性值的效果。calc() 函数可以使用加减乘除四种基本数学运算符来计算属性值&#xff0c;还可以使用括号来改变优先级。 二、ca…

【brew】Mac上安装vue3

先安装node。 这里我从其他博客找的方案&#xff0c;原始脚本下载太慢了。 cnpm的安装&#xff1a; 让npm更快一点。 npm install -g cnpm --registryhttps://registry.npm.taobao.org安装vue脚手架 2.0版本&#xff1a;sudo npm install -g vue-cli 3.0版本&#xff1a; sud…

Java实战案例————ATM

需求分析 首先ATM银行系统包括两个基础大功能&#xff1a;开户和登陆账户&#xff08;当然在系统中没有一个账户时不能登录&#xff0c;需要先开户&#xff09;。 一名用户有6项基本信息描述&#xff1a;姓名、性别、银行卡号、银行卡密码、账户余额、取款限额。 在登录账户…

QNX常用调试方法

QNX常用调试方法 1. top 查询系统状态最常用的工具是top&#xff0c;它可以显示系统资源的使用情况。我们最关心的通常是系统可用内存和CPU使用率。如果CPU使用率过高可能是因为某些应用存在bug&#xff0c;重点关注下面显示的占用CPU资源最多的几个线程。如果可用内存太少&am…