前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【C++11特性篇】一文带小白轻松理解【万能引用(引用折叠)】&【完美转发】

【C++11特性篇】一文带小白轻松理解【万能引用(引用折叠)】&【完美转发】

作者头像
YY的秘密代码小屋
发布2024-01-23 09:59:51
1.2K0
发布2024-01-23 09:59:51
举报
文章被收录于专栏:C++系列
在这里插入图片描述
在这里插入图片描述

一.万能引用

【1】基本概念

  • 万能引用 既可以接收左值,又可以接收右值
  • 实参是左值,他就是左值引用(引用折叠)
  • 实参是右值,他就是右值引用
  • PS:万能引用还有另一种叫法:引用折叠 ,就是当其传入参数为左值时,&&会折叠成&;当传入参数为右值时,&&不折叠照常接收

【2】在C++中的应用场景简述(代码演示)

  • 模板中的 && 不代表右值引用,而是 万能引用 ,其既能接收左值又能接收右值。
  • 模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的 能力
  • 但是引用类型的唯一作用就是—— 限制了接收的类型 ,后续使用中都退化成了 左值
代码语言:javascript
复制
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);     

 return 0;
}

二.完美转发

【1】完美转发应用的引入

根据本篇博客【第四part】中的结论: 右值引用变量的属性 会被编译器识别成 左值

  • 我们希望能够在传递过程中 保持它的左值或者右值的属性, 就需要用到 std::forward 完美转发

【2】基本概念

  • std::forward 完美转发 在传参的过程中保留 对象原生类型属性,即保持它的左值或者右值的属性

【3】在C++中的应用场景简述(代码演示)

代码语言:javascript
复制
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)
{
    // std::forward<T>(t)在传参的过程中保持了t的原生类型属性
	// 完美转发,t是左值引用,保持左值属性
	// 完美转发,t是右值引用,保持右值属性
    Fun(std::forward<T>(t));
}

int main()
{
 PerfectForward(10);           // 右值
 
 int a;
 PerfectForward(a);     

 return 0;
}

三.完美转发实际中的使用场景

【1】希望传入函数的右值能够保留右值走【移动构造】而不是【拷贝构造】

具体情景演示如下所示:

  • 比如在string的push_back函数中,以往,它是传左值,左值引用调用拷贝构造;
  • 而在C++11中,容器都支持了【移动构造】,所以我们有时会传右值,希望它能够走移动构造
  • 但问题是:右值引用变量的属性会被编译器识别成左值! 如下图中传到push_back函数中的val已经是左值了,想走匹配的右值引用实现移动构造就成了问题
  • 而我们此时给它用上【完美转发】在传参的过程中保留对象原生类型属性; val的属性仍然保存为右值,就可以正常走右值引用实现移动构造了
在这里插入图片描述
在这里插入图片描述

四.关于【左值引用】【右值引用】易混淆的知识点

【1】结论:右值引用变量的属性会被编译器识别成左值

  • 右值引用变量的属性会被编译器识别成左值
  • 否则在移动构造的场景下无法完成 资源转移(移动构造),必须要修改
在这里插入图片描述
在这里插入图片描述

【2】结论的证明(代码演示)

  • 我们可以观察下面代码,证明该结论:
代码语言:javascript
复制
int main()
{
    int a;
	int& r = a;
	int&& rr = move(a);//std::move()函数位于头文件中,该函数名字具有迷惑性,它并不搬移任何东西
	                    //唯一的功能就是将一个左值强制转化为右值引用,然后实现移动语义
	cout << &r << endl;
	cout << &rr << endl;  //我们知道右值不能取地址,不能被修改,而这里都能正常打印
	                     //证明结论:右值引用变量的属性会被编译器识别成左值

 return 0;
}
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-12-22,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一.万能引用
    • 【1】基本概念
      • 【2】在C++中的应用场景简述(代码演示)
      • 二.完美转发
        • 【1】完美转发应用的引入
          • 【2】基本概念
            • 【3】在C++中的应用场景简述(代码演示)
            • 三.完美转发实际中的使用场景
              • 【1】希望传入函数的右值能够保留右值走【移动构造】而不是【拷贝构造】
              • 四.关于【左值引用】【右值引用】易混淆的知识点
                • 【1】结论:右值引用变量的属性会被编译器识别成左值
                  • 【2】结论的证明(代码演示)
                  相关产品与服务
                  容器服务
                  腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
                  领券
                  问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档