C++11中std::forward的使用 (转)
std::forward argument: Returns an rvalue reference to arg if arg is not an lvalue reference; If arg is an lvalue reference, the function returns arg without modifying its type.
std::forward:This is a helper function to allow perfect forwarding of arguments taken as rvalue references to deduced types, preserving any potential move semantics involved.
std::forward<T>(u)有两个参数:T 与 u。当T为左值引用类型时,u将被转换为T类型的左值,否则u将被转换为T类型右值。如此定义std::forward是为了在使用右值引用参数的函数模板中解决参数的完美转发问题。
std::move是无条件的转为右值引用,而std::forward是有条件的转为右值引用,更准确的说叫做Perfect forwarding(完美转发),而std::forward里面蕴含着的条件则是Reference Collapsing(引用折叠)。
std::move不move任何东西。std::forward也不转发任何东西。在运行时,他们什么都不做。不产生可执行代码,一个比特的代码也不产生。
std::move和std::forward只是执行转换的函数(确切的说应该是函数模板)。std::move无条件的将它的参数转换成一个右值,而std::forward当特定的条件满足时,才会执行它的转换。
std::move表现为无条件的右值转换,就其本身而已,它不会移动任何东西。 std::forward仅当参数被右值绑定时,才会把参数转换为右值。 std::move和std::forward在运行时不做任何事情。
下面是从其他文章中copy的测试代码,详细内容介绍可以参考对应的reference:
- #include "forward.hpp"
- #include <utility>
- #include <iostream>
- #include <memory>
- #include <string>
- //////////////////////////////////////////////
- // reference: http://en.cppreference.com/w/cpp/utility/forward
- struct A {
- A(int&& n) { std::cout << "rvalue overload, n=" << n << "\n"; }
- A(int& n) { std::cout << "lvalue overload, n=" << n << "\n"; }
- };
- class B {
- public:
- template<class T1, class T2, class T3>
- B(T1&& t1, T2&& t2, T3&& t3) :
- a1_{ std::forward<T1>(t1) },
- a2_{ std::forward<T2>(t2) },
- a3_{ std::forward<T3>(t3) }
- {
- }
- private:
- A a1_, a2_, a3_;
- };
- template<class T, class U>
- std::unique_ptr<T> make_unique1(U&& u)
- {
- return std::unique_ptr<T>(new T(std::forward<U>(u)));
- }
- template<class T, class... U>
- std::unique_ptr<T> make_unique(U&&... u)
- {
- return std::unique_ptr<T>(new T(std::forward<U>(u)...));
- }
- int test_forward1()
- {
- auto p1 = make_unique1<A>(2); // rvalue
- int i = 1;
- auto p2 = make_unique1<A>(i); // lvalue
- std::cout << "B\n";
- auto t = make_unique<B>(2, i, 3);
- return 0;
- }
- ////////////////////////////////////////////////////////
- // reference: http://www.cplusplus.com/reference/utility/forward/
- // 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 test_forward2()
- {
- int a;
- std::cout << "calling fn with lvalue: ";
- fn(a);
- std::cout << '\n';
- std::cout << "calling fn with rvalue: ";
- fn(0);
- std::cout << '\n';
- return 0;
- }
- //////////////////////////////////////////////////////
- // reference: http://stackoverflow.com/questions/8526598/how-does-stdforward-work
- template<class T>
- struct some_struct{
- T _v;
- template<class U>
- some_struct(U&& v) : _v(static_cast<U&&>(v)) {} // perfect forwarding here
- // std::forward is just syntactic sugar for this
- };
- int test_forward3()
- {
- /* remember the reference collapsing rules(引用折叠规则):
- 前者代表接受类型,后者代表进入类型,=>表示引用折叠之后的类型,即最后被推导决断的类型
- TR R
- T& &->T& // lvalue reference to cv TR -> lvalue reference to T
- T& &&->T& // rvalue reference to cv TR -> TR (lvalue reference to T)
- T&& &->T& // lvalue reference to cv TR -> lvalue reference to T
- T&& &&->T&& // rvalue reference to cv TR -> TR (rvalue reference to T) */
- some_struct<int> s1(5);
- // in ctor: '5' is rvalue (int&&), so 'U' is deduced as 'int', giving 'int&&'
- // ctor after deduction: 'some_struct(int&& v)' ('U' == 'int')
- // with rvalue reference 'v' bound to rvalue '5'
- // now we 'static_cast' 'v' to 'U&&', giving 'static_cast<int&&>(v)'
- // this just turns 'v' back into an rvalue
- // (named rvalue references, 'v' in this case, are lvalues)
- // huzzah, we forwarded an rvalue to the constructor of '_v'!
- // attention, real magic happens here
- int i = 5;
- some_struct<int> s2(i);
- // in ctor: 'i' is an lvalue ('int&'), so 'U' is deduced as 'int&', giving 'int& &&'
- // applying the reference collapsing rules yields 'int&' (& + && -> &)
- // ctor after deduction and collapsing: 'some_struct(int& v)' ('U' == 'int&')
- // with lvalue reference 'v' bound to lvalue 'i'
- // now we 'static_cast' 'v' to 'U&&', giving 'static_cast<int& &&>(v)'
- // after collapsing rules: 'static_cast<int&>(v)'
- // this is a no-op, 'v' is already 'int&'
- // huzzah, we forwarded an lvalue to the constructor of '_v'!
- return 0;
- }
- ////////////////////////////////////////////////////
- // reference: https://oopscenities.net/2014/02/01/c11-perfect-forwarding/
- void sum(int a, int b)
- {
- std::cout << a + b << std::endl;
- }
- void concat(const std::string& a, const std::string& b)
- {
- std::cout<< a + b << std::endl;
- }
- void successor(int a, int& b)
- {
- b = ++a;
- }
- template <typename PROC, typename A, typename B>
- void invoke(PROC p, A&& a, B&& b)
- {
- p(std::forward<A>(a), std::forward<B>(b));
- }
- int test_forward4()
- {
- invoke(sum, 10, 20);
- invoke(concat, "Hello", "world");
- int s = 0;
- invoke(successor, 10, s);
- std::cout << s << std::endl;
- return 0;
- }
GitHub:https://github.com/fengbingchun/Messy_Test
C++11中std::forward的使用 (转)的更多相关文章
- C++11中std::forward的使用
std::forward argument: Returns an rvalue reference to arg if arg is not an lvalue reference; If arg ...
- C++11中std::move、std::forward、左右值引用、移动构造函数的测试
关于C++11新特性之std::move.std::forward.左右值引用网上资料已经很多了,我主要针对测试性能做一个测试,梳理一下这些逻辑,首先,左值比较熟悉,右值就是临时变量,意味着使用一次就 ...
- C++11中std::function的使用
class template std::function is a general-purpose polymorphic function wrapper. Instances of std::fu ...
- C++11中std::move的使用
std::move is used to indicate that an object t may be "moved from", i.e. allowing the effi ...
- C++11中std::bind的使用
std::bind: Each argument may either be bound to a value or be a placeholder: (1).If bound to a value ...
- C++11中std::unordered_map的使用
unordered map is an associative container that contains key-value pairs with unique keys. Search, in ...
- item 23: 理解std::move和std::forward
本文翻译自<effective modern C++>,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 根据std::move和std::forward不 ...
- 用C++11的std::async代替线程的创建
c++11中增加了线程,使得我们可以非常方便的创建线程,它的基本用法是这样的: void f(int n); std::thread t(f, n + 1); t.join(); 但是线程毕竟是属于比 ...
- C++11 使用 std::async创建异步程序
c++11中增加了线程,使得我们可以非常方便的创建线程,它的基本用法是这样的: void f(int n); std::thread t(f, n + 1); t.join(); 但是线程毕竟是属于比 ...
随机推荐
- bzoj 3622 已经没有什么好害怕的了——二项式反演
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3622 令 f[i] 表示钦定 i 对 a[ ]>b[ ] 的关系的方案数:g[i] 表 ...
- NPOI 创建Excel 设置宽度 样式 颜色对比表
前两天用NPOI来操作Office软件,在使用的时候有点问题,也有收获,就做个笔记 记录下来,主要做的事数据的导出功能.一些公共的方法,做个笔记. 更多的详细内容可以到NPOI的官方教程去看 htt ...
- push is not a function
push is not a function 今天写js 碰上了这个问题 找了半天百度不得其解. 后来发现,我push的是一个对象 push数组才可以
- HA 部署wordpress
前提: 1.保证免密认证ssh 2.NTP时间是否同步: 3.保证防火墙,selinux关闭: 4.用户名互相能够解析:在hosts文件设置: 环境: 系统:centos6.8和centos7.2 I ...
- 实验三:C分支结构程序设计
3.1 #include<stdio.h> int main(){ char a; scanf("%c",&a); if(a>=65&&a ...
- [转]NSIS:安装、卸载时检查程序是否正在运行
原文链接 http://www.flighty.cn/html/bushu/20110402_115.html 如果我们要安装或升级的程序正在运行,文件肯定会替换失败,以下代码可以提示用户结束正在运行 ...
- matplot 代码实例
matplot 代码实例 #!/usr/bin/env python # coding=utf-8 import numpy as np import matplotlib.pyplot as plt ...
- Ubuntu终端快捷键使用
快捷键 功能 Tab 自动补全 Ctrl+a 光标移动到开始位置 Ctrl+e 光标移动到最末尾 Ctrl+k 删除此处至末尾的所有内容 Ctrl+u 删除此处至开始的所有内容 Ctrl+d 删除当前 ...
- Centos 的计划任务 crontab
使用计划任务! crontab命令主要有3个参数: -e :编辑用户的crontab. -l :列出用户的crontab的内容. -r :删除用户的crontab的内容. 执行crontab -e,将 ...
- 学习git最好的方式
1:登陆git官网网站 https://git-scm.com 2:点击esay to learn连接 3:点击Book连接 4:选择简体中文,下载PDF文档,也可以在线学习.