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.hqwc.cn/news/125178.html

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

相关文章

多卡片效果悬停效果

效果展示 页面结构 从页面的结构上看&#xff0c;在默认状态下毛玻璃卡片是有层次感的效果叠加在一起&#xff0c;并且鼠标悬停在卡片区域后&#xff0c;卡片整齐排列。 CSS3 知识点 transform 属性的 rotate 值运用content 属性的 attr 值运用 实现页面整体布局 <div …

连续爆轰发动机

0.什么是爆轰 其反应区前沿为一激波。反应区连同前驱激波称为爆轰波。爆轰波扫过后&#xff0c;反应区介质成为高温高压的爆轰产物。能够发生爆轰的系统可以是气相、液相、固相或气-液、气-固和液-固等混合相组成的系统。通常把液、固相的爆轰系统称为炸药。 19世纪80年代初&a…

[论文必备]最强科研绘图分析工具Origin(2)——简单使用教程

本篇将介绍Origin的简单使用教程。 安装教程见上篇&#xff1a;[论文必备]最强科研绘图分析工具Origin&#xff08;1&#xff09;——安装教程 目录 &#x1f4e2;一、工具栏介绍 &#x1f4e3;1.1 行 1.1.1 标准栏 1.1.2 导入栏 1.1.3 工作表数据 1.1.4 图表数据 &a…

【代码阅读笔记】yolov5 rknn模型部署

一、main函数思路 二、值得学习的地方 1、关注yolov5检测流程 2、其中几个重要的结构体 typedef struct {int left;int right;int top;int bottom; } YOLOV5_BOX_RECT; // box坐标信息typedef struct {char name[YOLOV5_NAME_MAX_SIZE];int class_index;YOLOV5_BOX_RECT box…

实验三十五、LM117 稳压电源的设计

一、题目 利用 LM117 设计一个稳压电路&#xff0c;要求输出电压的调节范围为 5 ∼ 20 V 5\sim20\,\textrm V 5∼20V&#xff0c;最大负载电流为 400 mA 400\,\textrm{mA} 400mA。利用 Multisim 对所设计电路进行仿真&#xff0c;并测试所有性能指标。 二、仿真电路 仿真电…

【论文阅读】(CVPR2023)用于半监督医学图像分割的双向复制粘贴

目录 前言方法BCPMean-teacher and Traning StrategyPre-Training via Copy-PasteBidirectional Copy-Paste ImagesBidirectional Copy-Paste Supervisory Signals Loss FunctionTesting Phase 结论 先看这个图&#xff0c;感觉比较清晰。它整个的思路就是把有标签的图片和无标…

揭秘亚马逊、ebay自养号测评底层环境防关联技术

今天珑哥讲的是纯人工的自养号方式&#xff0c;已经在做过的人应该都懂&#xff0c;实现自养号所使用的IP和浏览器都有哪些 &#xff1f;都会有哪些问题。 一&#xff1a;市面上的IP有哪些&#xff1f;会遇到的问题&#xff1f; 922、luminati、googelfi、TM流量卡、Rola&…

大模型部署手记(3)通义千问+Windows GPU

1.简介 组织机构&#xff1a;阿里 代码仓&#xff1a;GitHub - QwenLM/Qwen: The official repo of Qwen (通义千问) chat & pretrained large language model proposed by Alibaba Cloud. 模型&#xff1a;Qwen/Qwen-7B-Chat-Int4 下载&#xff1a;http://huggingface…

WVP-28181协议视频平台搭建教程

28181协议视频平台搭建教程 安装mysql安装redis安装ZLMediaKit安装28181协议视频平台安装依赖下载源码编译静态页面打包项目, 生成可执行jar修改配置文件启动WVP 项目地址&#xff1a; https://github.com/648540858/wvp-GB28181-pro 说明: wvp-GB28181-pro 依赖redis和mysql中…

时序分解 | Matlab实现CEEMDAN完全自适应噪声集合经验模态分解时间序列信号分解

时序分解 | Matlab实现CEEMDAN完全自适应噪声集合经验模态分解时间序列信号分解 目录 时序分解 | Matlab实现CEEMDAN完全自适应噪声集合经验模态分解时间序列信号分解效果一览基本介绍程序设计参考资料 效果一览 基本介绍 Matlab实现CEEMDAN完全自适应噪声集合经验模态分解时间…

TCP端口崩溃,msg:socket(): Too many open files

一、现象 linux系统中运行了一个TCP服务器&#xff0c;该服务器监听的TCP端口为10000。但是长时间运行时发现该端口会崩溃&#xff0c;TCP客户端连接该端口会失败&#xff1a; 可以看到进行三次握手时&#xff0c;TCP客户端向该TCP服务器的10000端口发送了SYN报文&#xff0c;…

Kafka:介绍和内部工作原理

展示Kafka工作方式的简单架构。 什么是Kafka&#xff1f;为什么我们要使用它&#xff1f;它是消息队列吗&#xff1f; 它是一个分布式流处理平台或分布式提交日志。 Kafka通常用于实时流数据管道&#xff0c;即在系统之间传输数据&#xff0c;构建不断流动的数据转换系统和构…