首页 > 编程语言 >c++11:std::forward,完美转发

c++11:std::forward,完美转发

时间:2023-04-30 17:22:24浏览次数:50  
标签:11 std cout int void c++ && 转发

目录

1、不完美转发

2、完美转发

2.1、引用折叠

2.2、std::forward

1、不完美转发
所谓完美转发,是指在函数模板中,完全按照模板的参数的类型,将参数传递给函数模板中调用的另一个函数。比如:

template <typename T>
void IamForwording(T t)
{
IrunCodeActually(t);
}
上面的例子中,IamForwarding是一个转发函数模板。而函数IrunCodeActually则是真正执行代码的目标函数。对于目标函数IrunCodeActually而言,它总是希望转发函数将参数按照传入Iamforwarding时的类型传递(即传入ImaForwarding的是左值对象,IrunCodeActually就能获得左值对象,传入ImaForwarding的是右值对象,IrunCodeActually就能获得右值对象),而不产生额外的开销,就好像转发者不存在一样。

在ImaForwarding的参数中使用了最基本类型进行转发,该方法会导致参数在传给IrunCodeActually之前就产生了一次临时对象拷贝。因此这样的转发只能说是正确的转发,谈不上完美。

2、完美转发
C++11 标准中规定,通常情况下右值引用形式的参数只能接收右值,不能接收左值。但对于函数模板中使用右值引用语法定义的参数来说,它不再遵守这一规定,既可以接收右值,也可以接收左值(此时的右值引用又被称为“万能引用”)。

2.1、引用折叠
& + & -> &
& + && -> &
&& + & -> &
&& + && -> &&
一旦定义中出现了左值引用,引用折叠总是优先将其折叠为左值引用。

模板对类型的推导规则就比较简单了,当转发函数的实参是类型X的一个左值引用时,那么模板参数被推导为X&,而转发函数的实参是类型X的一个右值引用的话,那么模板的参数被推导为X&&类型。

结合以上的引用折叠规则,就能确定参数的实际类型,进一步,我们可以把转发函数写成如下形式:

template <typename T>
void IamForwording(T &&t)
{
IrunCodeActually(static_cast<T &&>(t));
}
当调用转发函数时传入了一个X类型的左值引用,转发函数被实例化为如下形式:

void IamForwording(X& &&t)
{
IrunCodeActually(static_cast<X& &&>(t));
}
应用上引用折叠规则,它就是:

template <typename T>
void IamForwording(X& t)
{
IrunCodeActually(static_cast<X& >(t));
}
这样一来,左值传递没有任何问题。但是就有人问了,你写static_cast是不是脑子有点问题啊,先别骂,接着往下看。

当我们调用转发函数传入一个X类型的右值引用时,转发函数被实例化为:

void IamForwording(X&& &&t)
{
IrunCodeActually(static_cast<X&& &&>(t));
}
应用上引用折叠规则,它就是:

void IamForwording(X&& t)
{
IrunCodeActually(static_cast<X&&>(t));
}
此处的static_cast起到的作用就是std::move的作用,因为std::move通常就是一个static_cast。不过在c++11中,用于完美转发的不是static_cast,也不是std::move,而是std::forward。

2.2、std::forward
std::forward和std::move在实际实现上差别不大,不过标准库这么设计,也许是为了让每个名字对应于不同的用途,以应对未来的扩展。

template <typename T>
void IamForwording(T&& t)
{
IrunCodeActually(std::forward<T>(t));
}
/*================================================================
* Copyright (C) 2022 baichao All rights reserved.
*
* 文件名称:forward.cpp
* 创 建 者:baichao
* 创建日期:2022年05月15日
* 描 述:
*
================================================================*/

// forward example
#include <utility> // std::forward
#include <iostream> // std::cout

// function with lvalue and rvalue reference overloads:
void overloaded(const int &x) { std::cout << "[lvalue]"; }
void overloaded(int &&x) { std::cout << "[rvalue]"; }

// function template taking rvalue reference to deduced type:
template <class T>
void fn(T &&x)
{
overloaded(x); // always an lvalue
overloaded(std::forward<T>(x)); // rvalue if argument is rvalue
}

int main()
{
int a;

std::cout << "calling fn with lvalue: ";
fn(a);
std::cout << '\n';

std::cout << "calling fn with rvalue: ";
fn(std::move(a));
std::cout << '\n';

return 0;
}
运行结果:

/*================================================================
* Copyright (C) 2022 baichao All rights reserved.
*
* 文件名称:forward2.cpp
* 创 建 者:baichao
* 创建日期:2022年05月15日
* 描 述:
*
================================================================*/

#include <iostream>

using namespace std;

void RunCode(int &&m) { cout << "rvalue ref" << endl; }
void RunCode(int &m) { cout << "lvalue ref" << endl; }
void RunCode(const int &&m) { cout << "const rvalue ref" << endl; }
void RunCode(const int &m) { cout << "const lvalue ref" << endl; }

template <typename T>
void PerfectForward(T &&t)
{
RunCode(forward<T>(t));
}

int main()
{
int a;
int b;
const int c = 1;
const int d = 0;

PerfectForward(a);
PerfectForward(move(b));
PerfectForward(c);
PerfectForward(move(d));
return 0;
}
运行结果:

 

/*================================================================
* Copyright (C) 2022 baichao All rights reserved.
*
* 文件名称:forward3.cpp
* 创 建 者:baichao
* 创建日期:2022年05月15日
* 描 述:
*
================================================================*/

#include <iostream>

using namespace std;

template <typename T, typename U>
void PerfectForward(T &&t, U &&func)
{
cout << t << "\tforwarded...";
func(forward<T>(t));
}

void RunCode(double &&m) { cout << "\tRunCode" << endl; }
void RunHome(double &&h) { cout << "\tRunHome" << endl; }
void RunComp(double &&c) { cout << "\tRunComp" << endl; }

int main()
{
PerfectForward(1.5, RunComp);
PerfectForward(8, RunCode);
PerfectForward(1.5, RunHome);
return 0;
}
运行结果:

希望我们都有重新开始的勇气!
————————————————
版权声明:本文为CSDN博主「_李白_」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_40179091/article/details/124784902

 

标签:11,std,cout,int,void,c++,&&,转发
From: https://www.cnblogs.com/im18620660608/p/17365477.html

相关文章

  • 11-react使用props.children 处理父子组件之间的传值
    //props.children组件传值import{Component}from"react"importreactDomfrom"react-dom"//床架一个createRef函数用来创建ref对象constHello=(props)=>{console.log(props)props.children('子组件传给父组件的值')//就是子组件标签之间的内容/......
  • C++ 基础语法
    C++基础语法基本输入输出#include<iostream>usingnamespacestd;intmain(){cout<<"xtloveac"<<endl;return0;}头文件#include<cstdio>包含printf(输出),scanf(输入)#include<iostream>包含cin(读入),cout(输出)usi......
  • C/C++ 各类型int、long、double、char、long long取值范围(基本类型的最大最小值)
    做题的时候经常会使用到数据类型的最大最小值(如int,long,longlong,char等),我也查了很多次,这次就记下来当笔记吧。参考了C++primeplus、各个博客、教程和c++官网,对C/C++中各个类型int、long、double、char、longlong等基本类型的取值范围即最大最小值总结如下:1字节=8位,......
  • C++之forward
    不管是T&&、左值引用、右值引用,std::forward都会按照原来的类型完美转发。forward主要解决引用函数参数为右值时,传进来之后有了变量名就变成了左值。 #include<QCoreApplication>#include<memory>#include<iostream>usingnamespacestd; template<typenameT>void......
  • 79.C++ 中的字面值常量
      一个形如42的值被称作字面值常量(literal),这样的值一望而知。每个字面值常量都对应一种数据类型,字面值常量的形式和值决定了它的数据类型。1.整型和浮点型字面值  可以将整型字面值写作十进制数、八进制数或十六进制数的形式。以0开头的整数代表八进制数,以0x或0X开头的代......
  • 【c++基础】程序运行时间计时
    使用std::chrono计算程序运行时间#include<iostream>#include<string>#include<chrono>voidRun(){ for(inti=0;i<1000000000;++i) { }}intmain(){ autobeforeTime=std::chrono::steady_clock::now(); Run(); autoafterTime......
  • VS Code 配置 C/C++ 环境(编译/调试)
    1,VSCode安装点击此处跳转到官网下载安装VSCode2,环境准备这里我们安装VSCode官方推荐的方法安装即可。官方文档。2.1下载MSYS2点击进入MSYS2官网,找到Installation的Downloadtheinstaller:msys2-x86_64-20230318.exe,点击msys2-x86_64-20230318.exe下载。然后根......
  • linux c/c++程序集成python库,实现调用python函数
    为了提高开发效率,扩展开发程序的功能,我们经常会在我们的linuxc/c++进程里调用外部脚本,例如lua、python,下面,介绍下如何在自己的linuxc/c++代码里调用python脚本里的函数和类,并且将python库集成到我们自己的进程目录里,这样就不依赖系统环境是否存在python及其版本要求。 ......
  • 11
    VS2022离线安装1.下载VS安装程序      ......
  • C++ STL容器
    vector变长数组,倍增的思想string字符串,substr(),c_str()queue队列,push(),front(),pop()priority_queue优先队列,push(),top(),pop()stack栈,push(),top(),pop()deque双端队......