当前位置: 首页 > news >正文

C++ 右值引用深入理解:特性、优化与底层

目录

一、左右值的概念及右值的种类

二、左值引用与右值引用

左值引用给右值取别名:

右值引用给左值取别名:

三、引用的意义及左值引用的场景

四、移动构造:

右值引用在底层里的实现:


一、左右值的概念及右值的种类

在 C++ 中,左右值的区分对于理解语言的行为至关重要。一般来说,可以取地址的是左值,不可以取地址的是右值。同时,被 const 修饰的通常也被视为左值。

右值主要包括以下几种类型:

  1. 常量,例如 10,它们在程序运行期间具有固定的值,且通常存储在只读内存区域。
  2. 临时对象,如 string("1111"),这些对象是在表达式中临时创建的,通常在包含它们的完整表达式结束后就可能被销毁。
  3. 匿名对象,例如 string("666"),没有名字的对象,同样也是临时创建的,生命周期较短。
	//右值10;//常量string("1111");//临时对象string("666");//匿名对象

二、左值引用与右值引用

  1. 左值引用给右值取别名

    • 不能直接用左值引用给右值取别名,但是使用 const 左值引用是可以的。这是因为 const 左值引用可以绑定到临时对象,确保在引用期间不会修改这个临时对象。例如 const int& a = 10;,这里 a 是一个常量引用,绑定到了右值 10
  2. 右值引用给左值取别名

    • 不能直接用右值引用给左值取别名,但是可以通过 std::move 将左值转换为右值后,再用右值引用进行引用。例如 int c = 100; int&& b = move(c); 和 int&& d = (int&&)c;
    • std::move 的作用是将左值转换为右值,本质上类似于强制类型转换。在底层实现中,无论是左值引用还是右值引用,实际上都是指针的形式,这也说明了在底层并没有真正的 “引用” 概念,而 move 更多的是在语法层面上进行类型转换。
int c = 100;
const int& a = 10;
int&& b = move(c);
int&& d = (int&&)c;

move:将左值转化为右值,本质等同于强转,所以上面的b与d没区别

move的讲解:move - C++ Reference (cplusplus.com)

三、引用的意义及左值引用的场景

引用在 C++ 中有重要的意义,主要是减少拷贝操作,提高程序的性能。

左值引用主要解决的场景包括引用传参和引用传返回值。通过引用传参,可以避免不必要的对象拷贝,特别是对于大型对象,这种方式可以显著提高程序的效率。在函数返回值时,如果返回左值引用,可以避免返回对象时的拷贝构造,直接返回对象的引用,提高程序性能。

然而,左值引用在某些情况下并没有彻底解决问题。例如,当传值返回的时候,如果返回的是左值,那么只能通过拷贝构造来解决问题。但是当遇到大数据结构(如 std::list 等)的深拷贝时,会非常浪费时间,这时候右值引用的价值就体现出来了。

四、移动构造

对于自定义类,可以定义移动构造函数来处理右值。例如对于 std::string 类,移动构造函数的形式可能是 string(string&& s),在这个函数中,通过 swap(s) 操作,将即将消亡的右值对象的数据交换过来,避免了不必要的拷贝操作。这样做的好处是,获取要消亡的数据,把自身不需要的数据给右值对象 s,让它去销毁,而自身获取有用的数据,不再需要进行拷贝。

string(string&& s)
{swap(s);
}

只有进行深拷贝的类才有移动构造的意义,因为只有在深拷贝的情况下,避免拷贝操作才能带来显著的性能提升。

右值返回种类:

存右值:内置类型右值

将亡值:类类型的右值

移动构造:swap(s),将要消亡的交换,获取要消亡的数据,把‘我’不要的数据给s,让他去销毁,而我获取有用的数据,不需要拷贝了吗

拷贝构造:传值返回时,编译器将要先把返回的数据先生成临时对象在拷贝给所需的对象优化成了直接把返回的数据拷贝给所需对象,将俩次拷贝优化成了一次

深拷贝的类才有移动构造的意义

在string s1 (s2)编译器开的优化大,构造加拷贝/移动构造 = 直接构造。相当于直接引用的main函数的对象—vs2022.不是所以编译器都这样  构造可以这么激进的优化,但是赋值不可以

移动赋值string s1 = (s2)

右值引用在底层里的实现:

R1右值引用的属性本身是---》左值  :因为swap什么的需要修改   

左值和右值可以来回切,本质只是语法层的限制

http://www.lryc.cn/news/464254.html

相关文章:

  • C# 文件操作
  • FFmpeg 4.3 音视频-多路H265监控录放C++开发三 :安装QT5.14.2, 并将QT集成 到 VS2019中。
  • Linux 累加计算递归算法汇编实现
  • 明日周刊-第23期
  • kubernets(二)
  • 《YOLO 标注工具全览》
  • 财富思维学习
  • python爬虫加解密分析及实现
  • 用Java做智能客服,基于私有知识库
  • 软考(网工)——网络安全
  • 如何给手机换ip地址
  • kafkamanager安装
  • 笔记本电脑U口保护分享
  • OpenCV高级图形用户界面(20)更改窗口的标题函数setWindowTitle()的使用
  • 结构体指针的初始化以及结构体变量作为函数实参传递时易混淆的知识点
  • Github 2024-10-20 php开源项目日报Top10
  • C++ 算法学习——1.3 双向深度优先搜索
  • Artistic Oil Paint 艺术油画着色器插件
  • 记一次left join联表查询的索引失效场景
  • 从零到一:前端开发者学习 Cocos Creator 的全攻略
  • JavaWeb 19 AJAX
  • element plus中menu菜单技巧
  • 数据结构-贪心算法笔记
  • 基于SpringBoot的在线汽车票预订平台
  • ubuntu 安装nginx
  • fanuc远程PNS启动
  • 使用 Spring 框架构建 MVC 应用程序:初学者教程
  • 集成Spring Security详解
  • Kettle9.4支持Clickhouse数据源插件开发以及性能测试
  • 微信支付V3 yansongda/pay 踩坑记录