万能引用、完美转发及其关系

万能引用

为什么需要万能引用。首先看下面三个函数模板,分别输入左值和右值,有哪些可以通过编译。

// 左值引用
template <typename T>
void fun(T& x){return ;}// 常量引用
template <typename T>
void fun1(const T& x){return ;}// 万能引用
template <typename T>
void fun2(T&& x){return ;}int main()
{int x = 1;fun(1);  // 错误,不接受右值fun(x);  // 正确,接受左值fun1(1);  // 正确,接受右值fun1(x);  // 正确,接受左值fun2(1);  // 正确,接受右值fun2(x);  // 正确,接受左值return 0;
}

可以得出以下结论:

  • 左值引用只能接受左值,不能接受右值;
  • 常量引用,既可以接受左值,又能接受右值。但是无法对引用对象进行修改。
  • 万能引用既可以接受左值,又能接受右值;可以对引用对象进行修改。

由此可以看出万能引用的灵活性。

那么万能引用推导出来的T是什么类型呢,请看以下代码。

#include <iostream>
using namespace std;// 用于打印T的类型
template <typename T>
std::string type_name()
{typedef typename std::remove_reference<T>::type TR;std::string r =  typeid(TR).name();if (std::is_const<TR>::value)r += " const";if (std::is_volatile<TR>::value)r += " volatile";if (std::is_lvalue_reference<T>::value)r += "&";else if (std::is_rvalue_reference<T>::value)r += "&&";return r;
}// 万能引用,打印推导结果T
template <typename T>
void printType(T&& x)
{cout << "Type of T is: " << type_name<T>() << endl;
}int main()
{cout << "------Test the type_name funciton------" << endl;cout << "Type is: " << type_name<int>() << endl;cout << "Type is: " << type_name<int&>() << endl;cout << "Type is: " << type_name<int&&>() << endl;int a = 1;int &lra = a;int &&rra = 1;cout << "------Input is r_value------" << endl;printType(1);printType(move(a));cout << "------Input is l_value------" << endl;printType(a);printType(lra);printType(rra);return 0;
}

输出如下:

------Test the type_name funciton------
Type is: i
Type is: i&
Type is: i&&
------Input is r_value------
Type of T is: i
Type of T is: i
------Input is l_value------
Type of T is: i&
Type of T is: i&
Type of T is: i&

第一部分说明type_name()这个函数能够正确打印出T的类型。

第二部分说明输入右值时,T被推导为原始类型,即T是int;这样printType(T&& x) 就变成了printType(int&& x),即x是右值引用。

第三部分说明输入左值时,T被推导为左值引用,即T是int&;这样printType(T&& x) 就变成了printType(int&&& x),折叠之后就是printType(int& x),即x是左值引用。

但需要注意的是,无论输入是左值还是右值,由于所输入内容已经绑定到了x上,也就是有了名字,因此x本身已经变成了左值。

重新阐述重点:万能引用中,输入右值,T被推导为不带引用的数据类型,如int;输入左值,T被推导为左值引用,如int&。

move函数的实现

在聊完美转发之前,先稍微介绍以下move函数的实现。

template <typename T>
typename remove_reference<T>::type&& move(T&& t)
{return static_case<typename remove_reference<T>::type&&>(t);
}

可见这就是一个万能引用,根据上一部分得出的规则,即当输入左值时,T被推导为左值引用类型(如int&),当输入右值,T被推导为不带引用的类型(如int)。

而typename remove_reference<T>::type 的作用是返回T去除引用后的类型,即无论输入左值还是右值,也即无论T被推导为是否带引用,该表达式得到的都是不带引用的数据类型。

从而move函数等价于(以int为例):

//将typename remove_reference<T>::type替换为int
template <typename T>
int&& move(T&& t)
{return static_case<int&&>(t);
}

也就是说无论输入左值还是右值,都将其转换为右值,这就是move函数的作用。为什么需要move函数呢,因为只有输入右值时,才会触发移动构造函数和移动赋值函数,所以就有了将左值变成右值的需求,而move函数就是干这个的。

完美转发及其实现

什么是转发:一个对象,传进第一个函数后,第一个函数又将其传入第二个函数进行处理,这就是对对象的转发。

什么是完美转发:传入第一个函数的是左值,那么被转发至第二个函数的也依然是左值;同理,传入第一个函数的是右值时,被转发至第二个函数的也仍然是右值。

完美转发必须依赖万能引用:假如没有万能引用,那么右值都无法传进第一个函数,更别提转发了。而且完美转发依赖于万能引用对T的推导。

如果实现完美转发:

// fun2是第二个函数
template<typename T>
void fun2(T& x){std::cout << "Lvalue ref" << std::endl;
}template<typename T>
void fun2(T&& x){std::cout << "Rvalue ref" << std::endl;
}// 万能引用,fun1这是第一个函数
template <typename T>
void fun1(T&& x)
{fun2(x);fun2(move(x));fun2(forward<T>(x));
}int main()
{int x = 1;cout <<"------Input Rvalue------"<<endl;fun1(1);cout <<"------Input Lvalue------"<<endl;fun1(x);return 0;
}/* output:
------Input Rvalue------
Lvalue ref
Rvalue ref
Rvalue ref
------Input Lvalue------
Lvalue ref
Rvalue ref
Lvalue ref
*/

上面尝试了三种方法:

第一种直接传递x。之前提到过,无论输入是左值还是右值,由于所输入内容已经绑定到了x上,也就有了名字,因此x本身已经变成了左值。所以当第一个函数传入右值时,直接传递的是左值,不是完美转发。

第二种是直接转递move(x)。然后此种情况下,当输入左值时,传递给第二个函数的变成了右值,不是完美转发。

第三种是使用forward<T>(x)函数,这是可以实现完美转发的。forward<T>()的原型有两个重载版本:

template <typename T>
T&& forward(typename std::remove_reference<T>::type& param)
{return static_cast<T&&>(param);
}template <typename T>
T&& forward(typename std::remove_reference<T>::type&& param)
{return static_cast<T&&>(param);
}

这里的T是在调用过程中显式指定的,又根据之前对typename std::remove_reference<T>::type的描述,无论T是啥,我们直接将其等价为无引用的原始数据类型,如int。

假如第一个函数输入的是右值,那么T就被推导为int,那么调用的就是forward<int>(x),对T和typename std::remove_reference<T>::type进行相应替换后,上面两个函数变成了下面的样子。又因为转发过程中,我们输入的x一定是左值,所以此时调用的是第一个函数,将输入强制转换为int&&并返回,即返回的仍然是右值。

// 将T替换为int
// 将typename std::remove_reference<T>::type替换为intint&& forward(int& param)
{return static_cast<int&&>(param);
}int&& forward(int&& param)
{return static_cast<int&&>(param);
}

假如第一个函数输入的是左值,那么T就被推导为int&,那么调用的就是forward<int&>(x)。同理可以得到函数的真实样子。x仍然一定是左值,还是调用第一个函数,将输入强制转换为int&&&,也即int&并返回,即返回的左值。

// 将T替换为int&
// 将typename std::remove_reference<T>::type替换为intint&&& forward(int& param)
{return static_cast<int&&&>(param);
}int&&& forward(int&& param)
{return static_cast<int&&&>(param);
}

综上,对于forward<T>(),其本质是根据T的推导对数据进行强制类型转换。

而完美转发的思路总结下来就是:

输入左值,万能引用中T推导为int&,则调用forward<int&>()把输入转换成了左值。

输入右值,万能引用中T推导为int,则调用forward<int>()把输入转换了右值。

这也就是为什么说完美转发需要依赖万能引用对T的推导的原因,只有结合万能引用和forward<T>()函数,才能实现完美转发。

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

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

相关文章

《Vite 基础知识》关于 .mjs .cjs 文件引出 NodeJS 对JS模块加载的思考(CommonJS 和 ESM)

前言 学习 Webpack/Vue2 升级 Vite/Vue3 时&#xff0c;发现以下不同&#xff1a; 新建的 Vitepress 项目默认创建了 config.mjs 文件&#xff1b; 新建的 Vite/Vue3 项目&#xff0c;package.json 中默认加上 type: module 配置&#xff1b; 新建的 Vite/Vue3 项目&#xf…

堆之大顶堆的介绍与实现

定义 堆是一棵完全二叉树&#xff0c;它大顶堆与小顶堆两类。 其中&#xff0c;大顶堆是指根结点比子结点均大的树。 创建 思路&#xff1a; 1.比较结点的左右孩子&#xff0c;记录较大值的下标。 2.将结点与该下标对应的值进行比较。若是孩子>结点&#xff0c;交换位置&…

【网络那些事】

【云计算】 云计算&#xff1a;把计算资源放在某个地方&#xff0c;并通过互联网暴露出来&#xff0c;让用户可以按需使用计算资源的方式&#xff0c;就是所谓的云计算 云计算的三种服务&#xff1a; 云平台专业名词 日常叫法 亚马逊云叫法 云服务器 ECS &#xff08;Elas…

循环结构:for循环,while循环,do-while,死循环

文章目录 for循环for案例&#xff1a;累加for循环在开发中的常见应用场景 whilewhile循环案例&#xff1a; for和while的区别&#xff1a;do-while三种循环的区别小结死循环 快捷键 ctrlaltt for循环 看循环执行多少次&#xff0c;就看有效数字有几个 快捷键 fori 示例代码&am…

Vuepress的使用

介绍 将markdown静态资源转换成html。 动态资源的转换还有很多&#xff0c;为什么要使用Vuepress&#xff1f; 目录分析 项目配置 详情 具体配置请看文档 插件配置 vuepress-theme-vdoing 主题插件 npm install vuepress-theme-vdoing -D先安装依赖配置主题 使用vuep…

MAC | linux | SSH 密钥验证

SSH密钥登陆过程 客户端通过ssh-keygen生成自己的公钥和私钥。手动将客户端的公钥放入远程服务器的指定位置。客户端向服务器发起 SSH 登录的请求。服务器收到用户 SSH 登录的请求&#xff0c;发送一些随机数据给用户&#xff0c;要求用户证明自己的身份。客户端收到服务器发来…

论文设计任务书学习文档|基于智能搜索引擎的图书管理系统的设计与实现

文章目录 论文(设计)题目:基于智能搜索引擎的图书管理系统的设计与实现1、论文(设计)的主要任务及目标2、论文(设计)的主要内容3、论文(设计)的基本要求4、进度安排论文(设计)题目:基于智能搜索引擎的图书管理系统的设计与实现 1、论文(设计)的主要任务及目标 …

报错:板端IP与PC的IP相同

报错&#xff1a; 配置 实际上我配置并没有错。 服务器IP&#xff08;就是本机&#xff09;、板端IP、网关。 解决 我网卡配置了多个IP。一番删除添加还是报错。 于是点击服务器IP&#xff0c;换成别的&#xff0c;再换回来&#xff0c;可以了&#xff1a;

架构面试题汇总:并发和锁(三)

在现代软件开发中&#xff0c;并发编程和多线程处理已成为不可或缺的技能。Java作为一种广泛使用的编程语言&#xff0c;提供了丰富的并发和多线程工具&#xff0c;如锁、同步器、并发容器等。因此&#xff0c;对于Java开发者来说&#xff0c;掌握并发编程和多线程处理的知识至…

【并发编程】深入理解ReentrantLock使用方法与原理解析(1)

引言 多线程编程中&#xff0c;为了保证线程安全&#xff0c;我们通常需要使用锁来协调对共享资源的访问。ReentrantLock&#xff08;可重入锁&#xff09;是Java提供的一种高级锁机制&#xff0c;相比于传统的synchronized关键字&#xff0c;它提供了更多的灵活性和控制。本文…

【大数据架构(3)】Lambda vs. Kappa Architecture-选择你需要的架构

文章目录 一. Data Processing Architectures1. Lambda Architecture1.1. 架构说明a. Data Ingestion Layerb. Batch Layer (Batch processing)c. Speed Layer (Real-Time Data Processing)d. Serving Layer 1.2. Lambda Architecture的优缺点1.3. 使用案例 2. Kappa Architect…

HTML+CSS+JS:花瓣登录组件

效果演示 实现了一个具有动态花朵背景和简洁登录框的登录页面效果。 Code <section><img src"./img/background.jpeg" class"background"><div class"login"><h2>Sign In</h2><div class"inputBox"…

重拾前端基础知识:CSS3

重拾前端基础知识&#xff1a;CSS3 前言边框圆角阴影图片 背景渐变文本字体多列动画与过渡2D 转换3D 转换过渡动画 网格布局弹性盒子&#xff08;重点&#xff09;父元素设置子元素设置 响应式设计设置 Viewport构建响应式网格视图12栅格媒体查询 案例讲解图片按钮分页 浏览器支…

python数据分析numpy基础之argmin求数组最小值索引

1 python数据分析numpy基础之argmin求数组最小值索引 python的numpy库的argmin()函数&#xff0c;用于获取沿指定轴的最小值的索引。 用法 numpy.argmin(a, axisNone, outNone, *, keepdims<no value>)描述 argmin()返回沿指定轴的最小值的索引。 入参axis表示指定轴…

【矩阵】计算矩阵边缘元素之和

每日一道算法题之计算矩阵边缘元素之和 一、题目描述二、思路三、C代码 一、题目描述 题目来源&#xff1a;洛谷 输入一个整数矩阵&#xff0c;计算位于矩阵边缘的元素之和。 所谓矩阵边缘的元素&#xff0c;就是第一行和最后一行的元素以及第一列和最后一列的元素。 C程序要求…

Windows 10中Visual Studio Code(VSCode)无法自动打开终端的解决办法

1.检查设置&#xff1a; 打开VSCode。点击左侧菜单栏的“文件”&#xff08;File&#xff09;。选择“首选项”&#xff08;Preferences&#xff09;。点击“设置”&#xff08;Settings&#xff09;。在搜索框中输入“shell”&#xff0c;然后点击“settings.json”进行编辑。…

学习鸿蒙基础(6)

一、Prop属性 父——>子 单向同步 Prop装饰的变量可以和父组件建立单向的同步关系。Prop装饰的变量是可变的&#xff0c;但是变化不会同步回其父组件。Prop装饰的变量和父组件建立单向的同步关系。Prop变量允许在本地修改&#xff0c;但修改后的变化不会同步回父组件。当父组…

【MATLAB】SVMD_ MFE_SVM_LSTM 神经网络时序预测算法

有意向获取代码&#xff0c;请转文末观看代码获取方式~也可转原文链接获取~ 1 基本定义 SVMD_MFE_SVM_LSTM神经网络时序预测算法结合了单变量分解&#xff08;SVMD&#xff09;、多尺度特征提取&#xff08;MFE&#xff09;、聚类后展开支持向量机&#xff08;SVM&#xff09;…

算法D31 | 贪心算法1 | 455.分发饼干 376. 摆动序列 53. 最大子序和

贪心算法其实就是没有什么规律可言&#xff0c;所以大家了解贪心算法 就了解它没有规律的本质就够了。 不用花心思去研究其规律&#xff0c; 没有思路就立刻看题解。 基本贪心的题目 有两个极端&#xff0c;要不就是特简单&#xff0c;要不就是死活想不出来。 学完贪心之后再…

rhcsa(rh134)

shell 查看用户shell a、如下查看/etc/shells文件列出了系统上所有可用的 shell&#xff08;具体的可用的 shell 列表可能会因不同的红帽版本和配置而有所不同&#xff09; &#xff08;如下图/etc/shells文件包含/bin/tmux并不意味着tmux是一个shell。实际上&#xff0c;/etc/…