C++中完美转发std::forward的一点点分析

2024-08-25 23:28

本文主要是介绍C++中完美转发std::forward的一点点分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前提
  • `std::forwar`是如何实现的?
  • `forwarder(a)`的调用过程
  • `forwarder(10)`的调用过程
  • `forwarder(std::move(a))`的调用过程
  • `std::move`是如何实现的
  • 总结

前提

之前在C++中的万能引用,引用折叠,完美转发中(如果你没有看过这篇文章,建议先去看看),主要介绍了万能引用,引用折叠的概念和使用,关于完美转发只是说了要这样用,但没有深入到源码底层去看看。这一节,我们把这个过程中使用的一些函数的源码分析下。

如分析有错误或不当之处,请大家多多见谅。

std::forwar是如何实现的?

先贴一段使用完美转发的代码。

#include <iostream>
#include <utility> // 包含 std::forward
#include <type_traits>// 普通函数重载,分别处理左值和右值
void process(int& x) {std::cout << "Processing lvalue: " << x << std::endl;
}void process(int&& x) {std::cout << "Processing rvalue: " << x << std::endl;
}// 泛型函数,用于转发参数
template <typename T>
void forwarder(T&& arg) {process(std::forward<T>(arg));
}int main() {![请添加图片描述](https://i-blog.csdnimg.cn/direct/13363849da724ad6bce3a39cc772202d.png)int a = 5;forwarder(a);    // 传递左值forwarder(10);   // 传递右值forwarder(std::move(a)); // std::move(a) 是一个右值引用return 0;
}

VS2017, C++标准为C++17的环境下,上述代码的输出结果为:

Processing lvalue: 5
Processing rvalue: 10
Processing rvalue: 5

接下来我准备采用从顶至下的方式,分别对这三个模板函数调用进行分析。

forwarder(a)的调用过程

模板函数forwarder使用了万能引用技术,所以它可以接收左值,也可以接收右值。所以forwarder(a)这样的调用完全合法。

在这里,我们省略了模板参数T,编译器会自己推导。那么第一个问题出现了,此时编译器推导的模板参数 T是什么?

答案:此时T会被推导为int&,如下所示:
请添加图片描述
然后你可以看到,forwader函数的定义变为了void forwarder(int & arg),这里为什么是int&呢?

答案:引用折叠。 因为 T=int&int &&&会发生引用折叠,变为int&

接着就到了process(std::forward<int &>(arg)),先来分析一下std::forward的源码,这样我们才能明白为什么传入左值可以调用左值版本的函数,传入右值可以调用右值版本的函数。

std::forward的源码如下:

template<class _Ty>_NODISCARD constexpr _Ty&& forward(remove_reference_t<_Ty>& _Arg) noexcept{	// forward an lvalue as either an lvalue or an rvaluereturn (static_cast<_Ty&&>(_Arg));}

是不是有点懵,别着急,我们将_Ty代进去就明白了,如下:
请添加图片描述
上面的分析已经比较清楚了,还有一个remove_reference_t留在下介绍。

接下来同理,我们分析下forwarder(10)的调用过程。

forwarder(10)的调用过程

请添加图片描述
forwarder(10)的调用过程和forwarder(a)一样,这里要注意的是T会被推导为int

forwarder(std::move(a))的调用过程

请添加图片描述
forwarder(std::move(a))forwarder(10)的调用过程一样,类型推导也一样,T=int

至此,关于std::forward的分析已经结束。那么我们来分析下:std::forward为什么可以实现完美转发。

简要分析:std::forward的底层就是一个static_cast<_Ty&&>,当_Ty不同时,static_cast出来的结果也不同,这决定了是调用process的左值引用版本,还是右值引用版本。那么_Ty是传入进来的模板参数,是在模板函数forwarder中捕获的。在这里_Ty = T,而T会根据我们传入的参数进行推导。

所以总的来说:当我们传入不同的参数时,模板就会拿到T,然后原封不动传给std::forwardstd::forward经过处理后,就可以调用不同版本的函数。

std::move是如何实现的

在文章的最后,我们来看下std::move的底层是如何实现的。源码如下:

请添加图片描述

总结

这一节我们分析了std::forward的调用流程,一步一步展示了其调用过程。
但是还有很重要的一点是:没有提出一个必须要使用std::forward的场景,也就是说,学了这么多,不知道在什么时候使用。等日后遇到了补上吧。

这篇关于C++中完美转发std::forward的一点点分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1106943

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

SWAP作物生长模型安装教程、数据制备、敏感性分析、气候变化影响、R模型敏感性分析与贝叶斯优化、Fortran源代码分析、气候数据降尺度与变化影响分析

查看原文>>>全流程SWAP农业模型数据制备、敏感性分析及气候变化影响实践技术应用 SWAP模型是由荷兰瓦赫宁根大学开发的先进农作物模型,它综合考虑了土壤-水分-大气以及植被间的相互作用;是一种描述作物生长过程的一种机理性作物生长模型。它不但运用Richard方程,使其能够精确的模拟土壤中水分的运动,而且耦合了WOFOST作物模型使作物的生长描述更为科学。 本文让更多的科研人员和农业工作者

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

MOLE 2.5 分析分子通道和孔隙

软件介绍 生物大分子通道和孔隙在生物学中发挥着重要作用,例如在分子识别和酶底物特异性方面。 我们介绍了一种名为 MOLE 2.5 的高级软件工具,该工具旨在分析分子通道和孔隙。 与其他可用软件工具的基准测试表明,MOLE 2.5 相比更快、更强大、功能更丰富。作为一项新功能,MOLE 2.5 可以估算已识别通道的物理化学性质。 软件下载 https://pan.quark.cn/s/57