C++11——神奇的右值引用与移动构造

文章目录

  • 前言
  • 左值引用和右值引用
  • 右值引用的使用场景和意义
  • 右值引用引用左值
  • 万能引用
  • 右值引用的属性
  • 完美转发
  • 新的默认构造函数
    • 强制和禁止生成默认函数
  • 总结

前言

本篇博客将主要讲述c++11中新添的新特性——右值引用和移动构造等,从浅到深的了解这个新特性的用法,以及它的意义是什么,并且最后深入探究了一些右值引用的细节问题, 并且从中引出了什么是万能引用和完美转发以及其的一些意义,会通过一些例子帮助大家更好的理解这一语法。

左值引用和右值引用

我们直到,传统的c++语法中就有引用的语法,而C++11中新增了右值引用的语法特性,所以从现在开始我们之前学习的引用都叫做右值引用,但首先要说明的一个点就是:无论是右值引用还是左值引用,都是在给对象取别名。

那么,如果想要了解右值引用到底有什么用,我们就需要知道什么是右值?右值和左值有什么区别?

左值: 一个表示数据的表达式(如变量名或者解引用的指针),我们可以获取它的地址,并且可以对它赋值。左值可以出现在赋值符号的左边,右值一般不可以出现在右值符号的左边。const修饰后的左值虽然不能赋值,但是可以取它的地址
如下是一些左值的例子:

int main() 
{// 以下的p、b、c、*p都是左值 int* p = new int(0); int b = 1; const int c = 2;// 以下几个是对上面左值的左值引用 int*& 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;fmin(x, y);// 以下几个都是对右值的右值引用 int&& rr1 = 10; double&& rr2 = x + y; double&& rr3 = fmin(x, y);//通过使用标准库中的move函数可以将左值转化成右值double&& rr4 = std::move(x);// 下面编译会报错:error C2106: “=”: 左操作数必须为左值 //10 = 1; //x + y = 1; //fmin(x, y) = 1;return 0;
}

通过以上定义我们就可以发现,一般区分左值和右值是通过是否能取地址来分辨的。

一个常量字符串(例“aabbcc”是左值还是右值呢?)
右值!虽然我们可以取得它的地址,但其并不是整个字符串的地址,而是字符串首元素的地址,并且它也不能放在赋值符号左边。

那么,了解了左值和右值的区别了之后,我们很容易发现,**const左值引用也是可以引用右值的!**既然如此,为什么又要创造右值引用这个新特性呢?如果想要解决这个问题,我们就要首先了解以下左值引用的缺陷并且引出右值引用的使用场景和意义了!


右值引用的使用场景和意义

为了更好的观察右值引用的意义所在,博主自己造了一个模仿c++string的简易轮子,这个模拟的string在进行构造和深拷贝的时候会打印信息,另外在这里还给出了移动拷贝和赋值的代码,进行移动拷贝和赋值也会打印信息(对于移动拷贝和赋值是什么后续会给出解答)

namespace lzz
{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) :_str(nullptr){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) noexcept:_str(nullptr), _size(0), _capacity(0){cout << "string(string&& s) -- 移动语义" << endl;swap(s);}// 移动赋值 string& operator=(string&& s) noexcept{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)string& operator+=(char ch){push_back(ch);return *this;}const char* c_str() const{return _str;}private:char* _str;size_t _size;size_t _capacity;};
};

首先,我们知道左值引用的诞生其实是为了提高程序运行的效率,做参数和做返回值是都可以提高效率!

void func1(lzz::string s) {}
void func2(const lzz::string& s) ()
int main()
{lzz::string s1("hello world");func1(s1);func2(s1);return 0;
}

在这里插入图片描述

注意,由于在模拟实现时深拷贝中调用了常量字符串构造函数,所以每次深拷贝时都会打印两次。
这极大的提高了大对象调用函数的效率。

左值引用的短板:
我们来想想左值引用还有什么短板,考虑当函数返回对象时一个局部变量时,当函数调用结束后,该变量就被销毁了,所以此时我们不能用左值引用返回,只能用传值引用返回。这样对性能是有比较大的损失的。

对于传值返回,较旧的编译器会进行两次拷贝构造,较新的编译器会进行优化,但也需要一次拷贝构造。

在这里插入图片描述


那么,此时右值引用就派生用场了。
右值引用和移动语义解决上述问题:
移动构造的本质是利用右值引用,将参数右值的资源窃取过来,占为己有,这样就不用做深拷贝了,就是窃取别人的资源来构造自己,移动赋值同理。

		// 移动构造 string(string&& s) :_str(nullptr), _size(0), _capacity(0){cout << "string(string&& s) -- 移动语义" << endl;swap(s);}

移动构造和移动赋值内部其实完成的是数据的交换工作,因为我们知道右值中的数据是会销毁的,如果不进行交换,就会导致在销毁时将需要保留的数据直接销毁,另外在进行移动赋值的时候还有可能导致原来的数据无法找到,造成内存泄漏

加上移动构造后上述代码的打印结果:

在这里插入图片描述

右值引用引用左值

按照语法,右值引用可以引用右值,但是右值引用可以引用左值吗?正常情况下是不可以的,但是标准库中为我们提供了一个函数move()可以做到这件事,传入左值之后,将会返回右值。

int main()
{lzz::string s1("xxx");//调用拷贝构造lzz::string s2(s1);//调用移动构造lzz::string s3(move(s1));
}

这里需要注意的是,虽然move()函数返回的是该对象的右值,但是仍然会修改该对象的数据,如下:
在这里插入图片描述

万能引用

右值引用出现的同时,c++还退出了一个模板的万能引用规则,语法如下:

template<typename T>
void func(T&& t)
{//...
}

也就是说模板中的&&并不是代表左值引用,而是万能引用,也就是说其既可以接受左值也可以接受右值!另外,如果传输的是对应const类型的引用,其也会自动推导成对应的const类型。

那么,在知道了万能引用之后,我们通过下面这些代码再次引出关于右值引用的一些问题。

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); }
int main() 
{PerfectForward(10); int a; PerfectForward(a);            // 左值 PerfectForward(std::move(a)); // 右值const int b = 8; PerfectForward(b);            // const 左值PerfectForward(std::move(b)); // const 右值 return 0;
}

可能有些人觉得,既然有万能引用,那肯定是穿什么类型调用什么类型的函数啊!这还不简单。

可是真有这么简单吗,我们先看结论,后解释结果。
在这里插入图片描述

结果既然是全是调用左值引用版本的func()!!这是为什么呢,难道是万能引用并不能引用右值吗?但是实际情况是万能引用确实可以引用左值,这里可以通过监视窗口进行查看:
在这里插入图片描述
这里我们传入10这个右值,可以发现此时t的类型是int&&,是右值引用,说明万能引用没有问题!那为什么会出现这种情况呢,这其实跟右值引用的属性有关,接下来我们就来深入探讨一下这一问题!

右值引用的属性

要探究这个问题,首先我们对PerfectForward(T&& t) 做一个修改,让其能够打印变量所在的地址

template<typename T> 
void PerfectForward(T&& t) 
{ cout << &t << endl;Fun(t); 
}

我们知道右值是不能打印地址的,所以这样子做正常来说应该会报错,但是呢结果如下:
在这里插入图片描述
没错,程序成功运行!并且打印出了所有地址!

这就说明了一个性质:右值引用的属性是左值!!!

其实,之所以这么搞,是为了右值引用更好的使用,想象一下,如果右值引用的属性是右值,那么我们就无法修改其中的内容,那么对于上面的移动构造和移动赋值,就无法进行swap()函数交换内容,也就无法提高效率,因此右值引用在这种场景下不能是右值。
const &&禁止了对其的修改权限

其实对于右值引用来说,对于将亡值而言可以理解为当其作为返回值返回时先暂时保留这快空间中的内容,并且能够修改这块空间,在使用结束后才真正销毁。

那么,有没有办法能够让参数保持当前的属性继续向下传递呢?答案是有的,c++考虑到了这个问题,所以设计了完美转发功能

完美转发

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

使用方法如下:

template<typename T> 
void PerfectForward(T&& t) 
{ Fun(std::forward<T>(t)); 
}

上面的代码加上完美转发之后,就可以属性原生属性的保持,打印结果如下:
在这里插入图片描述
完美转发的使用场景:

例如list<lzz::string> ls中,如果我们想要插入一个通过隐式类型转换的临时string,如ls.push_back("hello world"),我们想在push_back()内部调用string的移动构造函数,就必须使用完美转发,如下:
在这里插入图片描述

新的默认构造函数

原来的c++中,有6个默认成员函数

  1. 构造函数
  2. 析构函数
  3. 拷贝构造函数
  4. 拷贝赋值函数
  5. 取地址重载
  6. const取地址重载
    C++11新增两个:默认移动构造函数默认移动赋值函数

但这两个函数的生成条件是有限制的:

  • 如果你没有自己实现移动构造函数,且没有实现析构函数 、拷贝构造、拷贝赋值重载中的任 意一个(三个都没有实现)。那么编译器会自动生成一个默认移动构造。默认生成的移动构造函数,对于内置类型成员会执行逐成员按字节拷贝自定义类型成员,则需要看这个成员是否实现移动构造, 如果实现了就调用移动构造,没有实现就调用拷贝构造
  • 如果你没有自己实现移动赋值重载函数,且没有实现析构函数 、拷贝构造、拷贝赋值重载中 的任意一个(三个都没有实现),那么编译器会自动生成一个默认移动赋值。默认生成的移动构造函数,对于内 置类型成员会执行逐成员按字节拷贝,自定义类型成员,则需要看这个成员是否实现移动赋 值,如果实现了就调用移动赋值,没有实现就调用拷贝赋值。(默认移动赋值跟上面移动构造 完全类似
  • 如果你提供了移动构造或者移动赋值,编译器不会自动提供拷贝构造和拷贝赋值。

强制和禁止生成默认函数

C++11中还添加了一个强制生成默认函数的关键字default和禁止默认生成函数的关键字delete

如下:

class A
{
private:int _a;
public:~A() {}//提供了析构函数,不满足默认生成移动构造的性质A(A&& a) = default //强制自动生成//没有提供拷贝构造,禁止自动生成A(const A& a) = delete;
}

总结

以上就是关于C++11中右值引用的具体内容了,这段内容可以说是比较容易搞混,大家一定要自己去敲代码自己实验一下这些特性加深理解!另外,如果博主哪里写的有问题或者有疑惑的,欢迎评论区提出!

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

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

相关文章

数值分析学习笔记——绪论【华科B站教程版本】

绪论 数值分析概念 用计算机求解数学问题的数值方法和理论 三大科学研究方法 实验理论分析科学计算&#xff08;用计算机去辅助研究&#xff09;&#xff1a;数值方法计算机 解析解和近似解 解析解&#xff1a;使用数学方法求出或推导出的结果&#xff0c;往往可以求解出…

博途1200/1500 ALT指令

SMART PLC的ALT指令实现代码,请查看下面文章博客 SMART PLC如何构造ALT指令_smart200类似alt指令-CSDN博客单按钮启停这些老生常谈的问题,很多人感兴趣。这篇博文讨论下不同的实现方法,希望对大家有所帮助。指令虽然简单,但是在编程的时候合理使用对我们高效率编程帮助还是…

【算法学习】-【双指针】-【盛水最多的容器】

LeetCode原题链接&#xff1a;盛水最多的容器 下面是题目描述&#xff1a; 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。…

矢量图形编辑软件illustrator 2023 mac特点介绍

illustrator 2023 mac是一款矢量图形编辑软件&#xff0c;用于创建和编辑排版、图标、标志、插图和其他类型的矢量图形。 illustrator mac软件特点 矢量图形&#xff1a;illustrator创建的图形是矢量图形&#xff0c;可以无限放大而不失真&#xff0c;这与像素图形编辑软件&am…

react 网页/app复制分享链接到剪切板,分享到国外各大社交平台,通过WhatsApp方式分享以及SMS短信方式分享链接内容

1.需求 最近在做一个国际网站app,需要把app中某个页面的图文链接分享到国外各大社交平台上(facebook,whatapp,telegram,twitter等),以及通过WhatApp聊天方式分享&#xff0c;和SMS短信方式分享链接内容&#xff0c;该怎么做呢&#xff1f;图示如下: 分享到国外各大社交平台&am…

如何禁用Windows 10快速启动(以及为什么要这样做)

如果您不想启用Windows 10快速启动&#xff0c;则可以相对轻松地禁用它。 快速启动是一项功能&#xff0c;首先在 Windows 8 中作为快速启动实现&#xff0c;并延续到 Windows 10&#xff0c;让您的 PC 更快地启动&#xff0c;因此得名。虽然这个方便的功能可以通过将操作系统…

Linux——指令初识

Linux下基本指令 前言一、 ls 指令二、 pwd命令三、cd 指令四、 touch指令五、mkdir指令六、rmdir指令 && rm 指令七、man指令八、cp指令九、mv指令十、cat指令十一、.more指令十二、less指令十三、head指令十四、tail指令总结 前言 linux的学习开始啦&#xff01; 今…

手机或者电脑连接局域网内的虚拟机(网桥)

手机或者电脑连接局域网内的虚拟机&#xff08;网桥&#xff09; 手机软件&#xff1a;ConnectBot&#xff0c;Termius&#xff0c;JuiceSSH … 1.虚拟机vmware中添加桥接网卡 这里桥接网卡选择的是自动&#xff0c;是自动生成动态IP&#xff0c;如果不需要动态生成&#xff…

systemverilog function的一点小case

关于function的应用无论是在systemverilog还是verilog中都有很广泛的应用&#xff0c;但是一直有一个模糊的概念困扰着我&#xff0c;今天刚好有时间来搞清楚并记录下来。 关于fucntion的返回值的问题&#xff1a; function integer clog2( input logic[255:0] value);for(cl…

使用Jest测试Cesium源码

使用Jest测试Cesium源码 介绍环境Cesium安装Jest安装Jest模块包安装babel安装Jest的VSC插件 测试例子小结 介绍 在使用Cesium时&#xff0c;我们常常需要编写自己的业务代码&#xff0c;其中需要引用Cesium的源码&#xff0c;这样方便调试。此外&#xff0c;目前代码中直接使用…

ubuntu中的系统消息中显卡显示llvmpipe (LLVM 10.0.0, 256 bits)

这是我在使用ubuntu系统时出现的问题&#xff0c;网上搜到很多解决的办法&#xff0c;我是一顿操作&#xff0c;后来看到这位老哥的帖子解决了。 集Linux / Ubuntuwin10双系统安装记录(2):AMD核显驱动引发的问题 - 知乎上一篇中我们提到了 astroR2&#xff1a;Linux / Ubuntuw…

DataFrame入门

文章目录 1. 数据集合加载2. 使用常用的属性/方法查看数据情况type()shapecolumnsdtypesinfo() 3. 查看部分数据获取一列数据获取多列数据按行加载数据同时取出行列数据切片语法 4. 简单数据分析5. 数据可视化总结 1. 数据集合加载 pd.read_csv()方法不仅可以加载CSV文件&…

初识Java 12-3 流

目录 终结操作 将流转换为一个数组&#xff08;toArray&#xff09; 在每个流元素上应用某个终结操作&#xff08;forEach&#xff09; 收集操作&#xff08;collect&#xff09; 组合所有的流元素&#xff08;reduce&#xff09; 匹配&#xff08;*Match&#xff09; 选…

LLM下半场之Agent基础能力概述:Profile、Memory、Plan、Action、Eval学习笔记

一.Agent发展将会是LLM的下半场 目前大家都在讨论LLM&#xff0c;LLM解决的问题是帮助机器像人类一样理解彼此的意图&#xff0c;本质上来讲&#xff0c;LLM更像是一个技术或者工具。但是人类社会发生变革的引线&#xff0c;往往是一个产品或者解决方案&#xff0c;比如电池技…

Linux【网络】数据链路层

Linux【网络】数据链路层 数据链路层以太网帧格式对比理解MAC地址和IP地址ARP协议--地址解析协议ARP工作流程ARP请求ARP应答 其他协议DNS-域名解析协议ICMP--网络层协议NAT技术NAPT 正向代理与反向代理 数据链路层 数据链路层用于两个设备&#xff0c;同一数据链路节点之间的信…

栈和队列的实现

用栈实现队列 1.分析2.代码 1.分析 2.代码 class MyQueue {private Stack<Integer> s1;private Stack<Integer> s2;public MyQueue() {s1 new Stack<>();s2 new Stack<>();}public void push(int x) {s1.push(x);}public int pop() {if(empty()){re…

山西电力市场日前价格预测【2023-10-05】

日前价格预测 预测说明&#xff1a; 如上图所示&#xff0c;预测明日&#xff08;2023-10-05&#xff09;山西电力市场全天平均日前电价为363.87元/MWh。其中&#xff0c;最高日前电价为649.89元/MWh&#xff0c;预计出现在18: 45。最低日前电价为291.58元/MWh&#xff0c;预计…

微信公众号模板消息First,Remark字段不显示,备注字段不见了

今天在开发公众号过程中有个需求发模板消息我设置的如下 成绩单打印通知&#xff01;姓名&#xff1a;{{name.DATA}} 学号&#xff1a;{{stuid.DATA}}状态&#xff1a;{{status.DATA}}时间&#xff1a;{{date.DATA}} 备注&#xff1a;{{remark.DATA}} 然后发完通知发现《…

矩阵的c++实现(2)

上一次我们了解了矩阵的运算和如何使用矩阵解决斐波那契数列&#xff0c;这一次我们多看看例题&#xff0c;了解什么情况下用矩阵比较合适。 先看例题 1.洛谷P1939 【模板】矩阵加速&#xff08;数列&#xff09; 模板题应该很简单。 补&#xff1a;1<n<10^9 10^9肯定…

成都建筑模板批发市场在哪?

成都作为中国西南地区的重要城市&#xff0c;建筑业蓬勃发展&#xff0c;建筑模板作为建筑施工的重要材料之一&#xff0c;在成都也有着广泛的需求。如果您正在寻找成都的建筑模板批发市场&#xff0c;广西贵港市能强优品木业有限公司是一家值得关注的供应商。广西贵港市能强优…