解密C++中的forward<int>(a)和forward<int >(a):你真的了解它们之间的区别吗?

一文看尽C++中的forward完美转发

  • 一、前言
  • 二、深入理解forward和完美转发
  • 三、对`forward<int>(a)`的解析
  • 四、对`forward<int &&>(a)`的解析
  • 五、`forward<int>(a)`和`forward<int &&>(a)`的区别
  • 总结

一、前言

完美转发在C++中具有重要性,因为它允许函数将参数传递给其它函数,同时保持原始参数的值类别(左值或右值)和cv限定符(const或volatile)属性。这种机制对于实现通用的、灵活的代码至关重要,因为它可以确保函数接受的参数类型与其它函数调用相匹配,避免了不必要的参数类型转换。这种能力使得在编写泛型代码时,能够正确地传递参数并保持其原始属性,而不需要在函数调用链中添加多个重载或者模板函数来处理不同的参数类型。

在现代C++编程中,特别是在开发库和框架时,完美转发可以设计出更加灵活、通用的接口,提高代码的重用性和可维护性。同时还能够避免不必要的数据复制和额外的性能开销,提升代码的执行效率。因此,理解和正确应用完美转发是编写高效、高质量代码的关键。

本文的主旨是深入探讨C++中的forward<int>(a)forward<int &&>(a)之间的区别,以及它们在完美转发中的作用和应用。全面了解这两种形式的forward在C++中的具体用法、技术细节和差异,以及在实际开发中如何正确选择适当的形式进行参数传递和维护参数的值类别和cv限定符属性。

使用std::forward
入参
转发方法
目标对象

二、深入理解forward和完美转发

"forward"是C++语言中的一个重要概念,通常与"完美转发"相关联。在C++中,完美转发指的是在函数调用中保持参数的原始类型(左值或右值)和cv限定符(const或volatile)属性。使得函数可以将参数转发给其它函数,而不会丢失参数的原始属性

std::forward是C++标准库中的一个函数模板,用于在进行参数转发时保持参数的值类别和cv限定符。通常与模板参数的右值引用(T&&)结合使用,以实现完美转发。

完美转发和std::forward的主要用途

  1. 在实现通用函数时,允许函数将参数转发给其它函数,以保持参数的原始属性;
  2. 通过明晰地引入右值引用,避免因参数传递导致的不必要的数据拷贝,提高性能;
  3. 支持实现通用代码,以处理不同类型的参数和参数属性,实现更加灵活的解决方案。

在 C++ 中,每个表达式都有一个 值类别 和 一个 引用类别

  1. 值类别(Value Category):值类别描述了表达式产生的值的类别,以及该值是否可以被修改。C++ 中的值类别有两种:

    • lvalue:代表一个对象或者函数,可以取地址并且可以修改。
    • rvalue:代表一个临时对象,不可以取地址并且通常不可以被修改。
  2. 引用类别(Reference Category):引用类别描述了表达式的结果是一个引用还是一个值。C++ 中的引用类别有两种:

    • lvalue reference:产生一个 lvalue 引用,即可以被修改。
    • rvalue reference:产生一个 rvalue 引用,通常用于移动语义或者完美转发。

引用类别用于决定表达式的返回值是一个引用还是一个值,并且也与移动语义、完美转发等有关联。理解和使用值类别和引用类别可以更好地理解 C++ 中的对象生命周期、移动语义、函数重载匹配、完美转发等问题。

右值引用是实现完美转发的重要组成部分。右值引用是 C++11 中引入的一种引用类型,通过 && 符号声明。右值引用可以绑定到临时对象(即右值)或者具名的右值,而不能绑定到左值。右值引用在移动语义、完美转发等方面发挥着重要作用。

C++通过使用右值引用和模板参数,可以实现完美转发。标准库中提供了 std::forward 函数模板,它与右值引用结合使用,用于在函数调用中实现完美转发。

通过使用右值引用和 std::forward,可以在一个函数中将参数(包括左值和右值)完美地转发到另一个函数,同时保持参数的原始属性。使得函数可以接受任意类型的参数,并将其转发到其它函数,而不会失去参数的原始性质。

因此,右值引用为实现完美转发提供了一种重要的机制,通过结合使用右值引用和 std::forward,可以实现更加通用和灵活的函数模板,支持处理各种类型的参数并保持其原始属性。

三、对forward<int>(a)的解析

(1)forward<int>(a)的定义和用法。
std::forward 是一个模板函数,定义在 <utility> 头文件中。用于在函数模板中完美地转发参数,保持参数的原始类型和 cv 限定符属性。

函数原型:

template<class T>
constexpr T&& forward(typename std::remove_reference<T>::type& t) noexcept;

template<class T>
constexpr T&& forward(typename std::remove_reference<T>::type&& t) noexcept;

std::forward 接受一个模板参数 T 和一个参数 t,用于完美转发参数 t 的类型为 T。通过调用 std::forward,参数 t 的引用类型(左值引用或右值引用)可以被正确地转发到另一个函数,以保持参数的原始类型和属性。

forward<int>(a) 中的 <int> 是模板参数,用于指定希望将参数 a 转发的类型为 int。例如,当 a 是一个左值时,forward<int>(a) 将保持 a 的左值特性,将 a 以左值引用的形式转发到另一个函数中。

示例:

void process(int&& x) {
    // 处理右值 x
}

template<typename T>
void wrapper(T&& arg) {
    process(std::forward<T>(arg)); // 使用 std::forward 完美转发参数到 process 函数
}

int main() {
    int a = 5;
    wrapper(a); // 调用 wrapper 函数,将参数 a 以左值的形式转发到 process 函数
    return 0;
}

调用 wrapper(a) 时,参数 a 被转发到 process 函数,并保持了其原始类型和属性。在 wrapper 函数中,std::forward 用于完美转发参数,确保将左值引用转发给了 process 函数。

(2)参数传递的机制。
参数传递的机制指的是将参数传递给函数或方法时,参数在内存中是如何被处理和访问的。常见的参数传递机制包括值传递、引用传递和指针传递。

  1. 值传递(Pass by Value):在值传递机制中,函数的参数是通过将其值拷贝到函数内部来进行传递的。所以,在函数内对参数的修改不会影响到原来的值。值传递适用于传递基本数据类型和小型对象,但对于大型对象来说,由于需要进行复制,会带来一定的性能开销。

  2. 引用传递(Pass by Reference):在引用传递机制中,函数的参数是通过引用(即内存地址)进行传递的,而不是进行值的拷贝。所以,在函数内部对参数的修改会直接影响到原始值。引用传递适用于需要在函数内修改参数值的情况,同时也能避免额外的复制开销。

  3. 指针传递(Pass by Pointer):指针传递与引用传递类似,但是函数的参数是通过指针进行传递的。与引用不同的是,指针需要显式地进行解引用操作来获取参数值。指针传递通常用于需要可选参数或者需要在函数内修改参数指向对象的情况。

四、对forward<int &&>(a)的解析

(1)forward<int &&>(a)的定义和用法。函数原型:

template<class T>
constexpr T&& forward(typename std::remove_reference<T>::type& t) noexcept;

template<class T>
constexpr T&& forward(typename std::remove_reference<T>::type&& t) noexcept;

forward<int &&>(a)中,int && 表示正在使用模板函数 forward 来指示需要将参数 a 以右值引用的形式进行转发。

示例:

void process(int&& x) {
    // 处理右值 x
}

template<typename T>
void wrapper(T&& arg) {
    process(std::forward<T>(arg));  // 使用 std::forward 完美转发参数到 process 函数
}

int main() {
    int a = 5;
    wrapper(std::move(a));  // 调用 wrapper 函数,将参数 a 以右值引用的形式转发到 process 函数
    return 0;
}

通过std::move将参数 a 转换为右值,并将其作为参数传递给 wrapper 函数。然后,在 wrapper 中使用 std::forward 将参数以右值引用的形式转发到 process 函数中。这样就能在 process 函数中正确地处理右值引用参数。

(2)右值引用的特性:

  1. 右值引用可以绑定到临时对象(右值)。右值是指那些临时创建的、无法被引用的临时对象,例如函数返回值、临时对象、或者通过 std::move 转换的对象等。

  2. 可修改:通过右值引用可以对绑定的临时对象进行修改,并可以将其所有权(资源)转移给其他对象。这为实现移动语义提供了基础,使得在不需要进行深层拷贝的情况下可以高效地将对象传递给其他对象。

  3. 通过使用 std::forward 和模板推导,右值引用还可以用于实现完美转发(perfect forwarding),这样可以在函数模板中将参数以原始形式转发到其他函数,保持其原始类型和特性。

  4. 右值引用的引入可以实现移动语义,即将资源从一个对象“移动”到另一个对象,而不是进行昂贵的深层拷贝。对于动态分配的大型对象或资源管理类对象可以明显提高效率,例如 std::vector std::unique_ptr 等。

(3)对比 forward<int>(a)forward<int &&>(a)

  1. forward<int>(a):表示在使用 forward 函数模板时,要求参数 aint 类型进行转发。这会导致参数 a 被以左值引用类型进行转发。这样的转发方式适用于那些已经被声明为左值引用的变量或者表达式。

  2. forward<int &&>(a):表示要求参数 a 以右值引用类型进行转发。这样的转发方式适用于那些已经是右值引用的变量或者表达式,或者希望将参数以右值引用的形式进行转发,以便实现移动语义或者完美转发。

(4)应用场景:

  • 需要在函数模板中保持参数的原始类型和 cv 限定符属性,并且原始参数是一个左值引用时,使用 forward<int>(a) 来确保以相同的类型进行转发。

  • 需要在函数模板中对右值引用进行转发,保持其原始类型和 cv 限定符属性,并且假设原始参数是一个右值引用时,使用 forward<int &&>(a)。这在实现移动语义、完美转发等情况下非常有用。

(5)知识扩展: forward<int&>(a) 。使用 forward<int&>(a) 时要求参数 aint & 类型进行转发会导致参数 a 被以左值引用类型进行转发。这样的转发方式适用于那些已经是左值引用的变量或者表达式。原始参数是一个左值引用时,使用 forward<int&>(a) 来确保以相同的类型进行转发。

五、forward<int>(a)forward<int &&>(a)的区别

forward<int>(a)forward<int &&>(a) 的区别在于参数类型引用折叠以及模板参数推导上的处理方式。

  1. 参数类型:

    • forward<int>(a) 表示要求参数 aint 类型进行转发。即参数 a 被以左值引用类型进行转发。
    • forward<int &&>(a) 表示要求参数 aint && 类型进行转发。即参数 a 被以右值引用类型进行转发。
  2. 引用折叠:

    • 当参数 a 是一个左值时,forward<int>(a) 中的引用折叠会使参数 a 被以左值引用类型进行转发。
    • 当参数 a 是一个右值时,forward<int &&>(a) 中的引用折叠会使参数 a 被以右值引用类型进行转发。

forward<int>(a) 情况下编译器将使用模板参数推导,挑选出与 int 最匹配的类型,即值类型引用。也就是说,如果a是一个左值,它会被转发为一个左值引用,如果a是一个右值,它会被转发为一个右值引用。

forward<int &&>(a)使用了特殊的类型推导,会将参数a转发为右值引用。无论a是左值还是右值,它都会被转发为一个右值引用。

示例:

#include <iostream> 
using namespace std;

template <class T> 
void Print(T &t) 
{ 
	cout << "L" << t << endl; 
}
template <class T> 
void Print(T &&t) 
{ 
	cout << "R" << t << endl; 
}
template <class T> 
void func(T &&t) 
{ 
	Print(t); 
	Print(std::move(t)); 
	Print(std::forward<T>(t)); 
}

int main() {
	cout << "-- func(1)" << endl; 
	func(1); 
	int x = 10; 
	int y = 20; 
	cout << "-- func(x)" << endl; 
	func(x); // x本身是左值 
	cout << "-- func(std::forward<int>(y))" << endl; 
	func(std::forward<int>(y)); 

	cout << "-- func(std::forward<int&>(y))" << endl;
	func(std::forward<int&>(y));
	return 0; 
}

执行结果:

-- func(1)
L1
R1
R1
-- func(x)
L10
R10
L10
-- func(std::forward<int>(y))
L20
R20
R20
-- func(std::forward<int&>(y))
L20
R20
L20

总结

在C++中,forward<int>(a)forward<int &&>(a) 都是使用完美转发(forwarding)的技术,用于保持参数的值类别(左值或右值)。

  • forward<int>(a) 使用了模板参数推导,会将 aint 类型进行转发。无论传入的参数 a 是左值还是右值,它会保持其原始的值类别(即保持左值或右值属性)。

  • forward<int &&>(a) 也使用模板参数推导,但此时指定了参数类型为右值引用。

在这两种情况下,使用了std::forward,它在完美转发的过程中保留了传入参数的值类别,并将其正确地转发给新的函数。这在实现泛型代码时非常重要,能够确保正确地处理左值引用和右值引用。
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

vmware安装openEuler操作系统

vmware安装openEuler操作系统 1、下载openEuler操作系统镜像文件2、安装openEuler操作系统3、配置使用openEuler操作系统 1、下载openEuler操作系统镜像文件 官网下载链接 链接: https://www.openeuler.org/zh/download/ 这里选择 openEuler 22.03 LTS SP2 版本 标准镜像包 文…

Dungeon Scrawl——在线虚拟世界地图制作器

今天带来一款十分有趣的地图应用&#xff0c;同样也是在线地图工具&#xff0c;如果我们想要制作此类风格的地图&#xff0c;这款工具定能助我们一臂之力。 按照惯例先给出网址&#xff1a; Dungeon Scrawl | Free Online D&D Map Maker 进入网站&#xff0c;点击Start S…

Post-hoc Concept Bottleneck Models (PCBM)

ICLR 2023 spotlight 文章链接&#xff1a;https://arxiv.org/abs/2205.15480 代码链接&#xff1a;https://github.com/mertyg/post-hoc-cbm 一、概述 Post-hoc CBM&#xff08;PCBM&#xff09;也是CBM大家族中的一员&#xff0c;因此它的基本逻辑与CBM一致&#xff0c;就是…

从登录测试谈测试用例

谈谈登录测试&#xff1a; 可能你会说&#xff0c;“用户登录”这个测试对象也有点太简单了吧&#xff0c;我只要找一个用户&#xff0c;让他在界面上输入用户名和密码&#xff0c;然后点击“确 认”按钮&#xff0c;验证一下是否登录成功就可以了。的确&#xff0c;这构成了一…

C/C++ 函数的默认参数

下面介绍一项新内容 - 默认参数。 默认参数指的是当函数调用中省略了实参时自动使用的一个值。 例如&#xff0c;如果将 void wow (int n)设置成n 有默认值为1&#xff0c;则函数调用 wow()相当于 wow(1)这极大地提高了使用函数的灵活性。 假设有一个名为left()的函数&#xff…

构建安全的SSH服务体系

某公司的电子商务站点由专门的网站管理员进行配置和维护&#xff0c;并需要随时从Internet进行远程管理&#xff0c;考虑到易用性和灵活性&#xff0c;在Web服务器上启用OpenSSH服务&#xff0c;同时基于安全性考虑&#xff0c;需要对 SSH登录进行严格的控制&#xff0c;如图10…

记一次JSF异步调用引起的接口可用率降低 | 京东云技术团队

前言 本文记录了由于JSF异步调用超时引起的接口可用率降低问题的排查过程&#xff0c;主要介绍了排查思路和JSF异步调用的流程&#xff0c;希望可以帮助大家了解JSF的异步调用原理以及提供一些问题排查思路。本文分析的JSF源码是基于JSF 1,7.5-HOTFIX-T6版本。 起因 问题背景…

【MATLAB】【数字信号处理】基本信号的仿真与实现

目的 1、用MATLAB软件实现冲激序列 2、用MATLAB软件实现阶跃序列 3、用MATLAB软件实现指数序列 4、用MATLAB软件实现正弦序列 内容与测试结果 1、用MATLAB软件实现冲激序列 程序如下&#xff1a; % 1 冲激序列 clc; clear all; n0 -10; nf 50; ns 1; A 1;%起点为-1&…

SpringBoot灵活集成多数据源(定制版)

如来说世界&#xff0c;非世界&#xff0c;是名世界 如来说目录&#xff0c;非目录&#xff0c;是名目录 前言前期准备代码实现演示扩展 前言 本篇博客基于SpringBoot整合MyBatis-plus&#xff0c;如果有不懂这个的&#xff0c; 可以查看我的这篇博客&#xff1a;快速CRUD的秘诀…

Linux 权限掌控术:深入探索和用户管理

文章目录 前言1.外壳程序是什么&#xff1f;外壳程为什么存在&#xff1f;工作原理外壳程序怎么个事&#xff1f; 2. Linux权限的概念2.1 什么是权限2.2权限的本质2.3 Linux中的用户 3. 普通用户变成rootlinux中有三种人 4.Linux中文件的权限4.1文件的属性权限4.2 掌握修改权限…

数字集成系统设计——逻辑综合

目录 一、概述 1.1 综合的分类 1.2 逻辑综合的基本架构 1.3 逻辑综合的内部流程 1.3.1 RTL代码转译&#xff08;Translation&#xff09; 1.3.2 逻辑级优化&#xff08;Optimization&#xff09; 1.3.3 工艺映射&#xff08;Mapping&#xff09; 二、优化策略 2.1 资源…

Linux之进程管理

什么是进程 在linux中每个执行的程序都称为一个进程&#xff0c;每个进程都分配一个ID号&#xff08;pid进程号&#xff09;。每个进程都可能以两种方式存在&#xff0c;即前台和后天。前台进程就是用户目前的屏幕上可以进行操作的。后台进程则是实际在操作&#xff0c;但屏幕…

AD教程 (二十一)模块化布局规划

AD教程 &#xff08;二十一&#xff09;模块化布局规划 原理图是按照我们的功能模块去进行排布划分的 利用交叉选择模式分屏快速进行模块化布局 分屏&#xff0c;选中任意文档&#xff0c;右击&#xff0c;点击垂直分割 交叉选择模式&#xff0c;点击工具&#xff0c;交叉选…

【模拟电路】软件Circuit JS

一、模拟电路软件Circuit JS 二、Circuit JS软件配置 三、Circuit JS 软件 常见的快捷键 四、Circuit JS软件基础使用 五、Circuit JS软件使用讲解 欧姆定律电阻的串联和并联电容器的充放电过程电感器和实现理想超导的概念电容阻止电压的突变&#xff0c;电感阻止电流的突变LR…

基于SpringBoot的校园二手闲置交易平台

基于SpringBoot的校园二手闲置交易平台的设计与实现~ 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBootMyBatis工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 主页 登录界面 管理员界面 摘要 本文基于Spring Boot框架设计并实现了一款…

buuctf-Misc 题目解答分解103-105

103.[GKCTF 2021]签到 追踪流发现类似flag 字符 f14g 下面有大量的是16进制字符 64306c455357644251306c6e51554e4a5a3046355355737764306c7154586c4a616b31355357704e65556c7154586c4a616b31355357704e65556c7154586c4a616b31355357704e65556c7154586c4a616b31355357704e655…

git rebase应用场景三

文章目录 git rebase应用场景三 git rebase应用场景三 在我们的开发分支中 假设我们修改一个文件 提交一个版本 再回到master分支 同时也去修改1.txt文件&#xff0c;提交一个版本 这样相当于master分支提交了一次&#xff0c;dev也提交了一次 然后回到dev分支 此时会报错…

【网络安全】upload靶场pass11-17思路

目录 Pass-11 Pass-12 Pass-13 Pass-14 Pass-15 Pass-16 Pass-17 &#x1f308;嗨&#xff01;我是Filotimo__&#x1f308;。很高兴与大家相识&#xff0c;希望我的博客能对你有所帮助。 &#x1f4a1;本文由Filotimo__✍️原创&#xff0c;首发于CSDN&#x1f4da;。 &#x…

gRPC之内置Trace

1、内置Trace grpc内置了客户端和服务端的请求追踪&#xff0c;基于golang.org/x/net/trace包实现&#xff0c;默认是开启状态&#xff0c;可以查看事 件和请求日志&#xff0c;对于基本的请求状态查看调试也是很有帮助的&#xff0c;客户端与服务端基本一致&#xff0c;这里…

Delphi6函数大全4-SysUtils.pas

Delphi6函数大全4-SysUtils.pas首部 function FormatFloat(const Format: string; Value: Extended): string; $[SysUtils.pas功能 返回浮点数类型以指定格式字符串Format转换成字符串说明 FormatFloat(,.00, 1234567890) 1,234,567,890.00参考 function …
最新文章