C++11中的完美转发

news/2024/11/23 4:00:11/

C++11中的完美转发

在讨论引用折叠这个话题之前,先回顾一下C++11中的引用,

在C++11中引用有4种:非常量左值引用非常量右值引用常量左值引用常量右值引用。其中常量右值引用没有应用价值,所以我们不考虑。

  1. 非常量左值引用只能绑定非常量左值
  2. 非常量右值引用只能绑定非常量右值
  3. 常量左值引用又称万能引用,他能绑定所有值,但是不能修改被绑定的值
  4. 常量右值引用能绑定右值,但是不能修改被绑定的值(没有实用价值)

我们在模板中经常会调用另一个模板函数或者函数,我们希望我们能够在传参时,减少临时对象的产生,即我们总是希望按照引用传参,这时我们就提出了完美转发的概念。

template <class T>
void foo(T t)
{run(t);
}

1.引用折叠

typedef const int T;
typedef T& TR;
typedef TR&& TRRR;
TRRR&& a=1;

上面这段代码在C++98中是无法编译的, 而在C++11中却是可以通过的,TRRR &&的类型会被推导为const int &

下面给出具体的折叠规则:

typedef T& TR;
TR v;//T&
TR &v;//T&
TR && v;//T&typedef T&& TR;
TR v;//T&&
TR & v;//T&
TR && v;//T&&

即左值引用和右值引用在折叠的时候,优先折叠为左值引用

2.完美转发

#include<iostream>
using namespace std;
void run(int &&m)
{cout<<"rvalue reference\n";
}
void run(int &m)
{cout<<"lvalue reference\n";
}void run(const int &m)
{cout<<"const lvalue reference\n";
}
void run(const int &&m)
{cout<<"const rvalue reference\n";
}template <class T>
void foo( T &&t)
{run(static_cast<T&&>(t));
}
int main()
{int a;foo(a);foo(move(a));const int b=1;foo(b);foo(move(b));}
lvalue reference
rvalue reference
const lvalue reference
const rvalue reference

我们看一下,这个模板函数foo中实现的就是完美转发

template <class T>
void foo( T &&t)
{run(static_cast<T&&>(t));
}

如果a是个X类型的左值,那么T会被推导为X&,则模板实例化为

void foo( X& && t)
{run(static_cast<X& &&>(t));
}

根据引用折叠理论,上面就会变成左值引用

而如果a是个右值,那么T就会被推导为X,则模板实例化为

void foo( X&& t)
{run(static_cast<X&&>(t));
}

所以不管我们传递的是左值还是右值,run中都是按照引用传递的,这就是完美转发。实际上,真正奏效的是static_cast<T &&>()函数,它根本解决的是右值引用本身是个左值,所以得把他转化成右值这件事,
我们会发现,它做的事情和std::move()是一样的事情,但是为了后续区分,我们一般实用forward<T>()函数来代替static_cast<T&&>
注意,move()forward<T>()的区别在于,move是无条件转化为右值,而forward是当且仅当它是右值引用变成的,而且我们要提供模板参数T,用以告知参数的原来类型。

完美转发的应用在于包装函数

#include<iostream>
using namespace std;void func1(int &&)
{cout<<"func1"<<endl;
}
void func2(int &&)
{cout<<"func2"<<endl;
}template <class T,class U>
void foo( T &&t, U& func)
{func(forward<T>(t));
}
int main()
{foo(1,func1);foo(2,func2);
}

上面这种设计就是封装多个函数为一个函数,在C++11中make_pairmake_unique都是用上面这种方式实现的


http://www.ppmy.cn/news/19937.html

相关文章

编译原理学习笔记14——属性文法与语法制导翻译1

编译原理学习笔记14——属性文法与语法制导翻译114.1 属性文法14.2 属性计算14.1 属性文法 属性文法 综合属性 自下而上传递信息语法规则&#xff1a;根据右 部候选式中的符号 的属性计算左部被 定义符号的综合属性语法树&#xff1a;根据子结 点的属性和父结点 自身的属性…

《C++程序设计原理与实践》笔记 第11章 定制输入/输出

在本章中&#xff0c;我们重点关注如何使第10章中介绍的通用iostream框架适配特定的需求和偏好。 11.1 规则性和不规则性 C标准库的输入/输出部分——iostream库为文本的输入和输出提供了一个统一的、可扩展的框架。 到目前为止&#xff0c;我们将所有输入源视为等价的&…

网络原理之HTTP/HTTPS、TCP、IP四层协议栈

文章目录一、应用层&#xff08;一&#xff09;xml协议&#xff08;二&#xff09;json协议&#xff08;三&#xff09;protobuffer协议&#xff08;四&#xff09;HTTP协议1. 抓包工具&#xff0c;fiddler2. HTTP报文格式3. HTTP请求(Request)&#xff08;1&#xff09;URL基本…

2023需要重点关注的四大AI方向

2023需要重点关注的四大AI方向 过去10年&#xff0c;人工智能从实验室走向各行各业&#xff0c;成为一种普遍技术应用于众多领域。根据IDC的数据&#xff0c;2022年全球AI市场规模达到4328亿美元&#xff0c;增长近20%。而Precedence Research预计&#xff0c;到2030年&#xf…

一、Swagger简介

一、简介是springfox-swagger的增强UI实现&#xff0c;为Java开发者在使用Swagger的时候&#xff0c;能拥有一份简洁、强大的接口文档体验 效果&#xff1a;http://swagger-bootstrap-ui.xiaominfo.com/doc.html 示例:swagger-bootstrap-ui-demo: knife4j 以及swagger-bootst…

Modbus协议完整版

第一部分&#xff1a;Modbus协议1 引言1.1 范围MODBUS是OSI模型第7层上的应用层报文传输协议&#xff0c;它在连接至不同类型总线或网络的设备之间提供客户机/服务器通信。自从1979年出现工业串行链路的事实标准以来&#xff0c;MODBUS使成千上万的自动化设备能够通信。目前&am…

Linux常见的进程间通信

目录管道pipe匿名管道接口介绍示例代码fifo命名管道接口介绍代码示例匿名管道与命名管道的区别shm共享内存接口介绍相关指令代码示例特点总结信号信号量socket套接字管道 管道是一种较老的&#xff0c;半双工通信方式&#xff0c;即数据只能向一个方向流动&#xff08;即一个进…

Java---微服务---分布式搜索引擎elasticsearch(1)

分布式搜索引擎elasticsearch&#xff08;1&#xff09;1.elasticsearch1.1.了解ES1.1.1.elasticsearch的作用1.1.2.ELK技术栈1.1.3.elasticsearch和lucene1.1.4.为什么不是其他搜索技术&#xff1f;1.1.5.总结1.2.倒排索引1.2.1.正向索引1.2.2.倒排索引1.2.3.正向和倒排1.3.es…