C++11中的完美转发
创始人
2024-05-19 10:06:10
0

C++11中的完美转发

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

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

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

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

template 
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
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 
void foo( T &&t)
{run(static_cast(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 
void foo( T &&t)
{run(static_cast(t));
}

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

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

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

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

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

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

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

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

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

上一篇:Python调用lua

下一篇:TII投稿时间线

相关内容

热门资讯

中证A500ETF摩根(560... 8月22日,截止午间收盘,中证A500ETF摩根(560530)涨1.19%,报1.106元,成交额...
A500ETF易方达(1593... 8月22日,截止午间收盘,A500ETF易方达(159361)涨1.28%,报1.104元,成交额1...
何小鹏斥资约2.5亿港元增持小... 每经记者|孙磊    每经编辑|裴健如 8月21日晚间,小鹏汽车发布公告称,公司联...
中证500ETF基金(1593... 8月22日,截止午间收盘,中证500ETF基金(159337)涨0.94%,报1.509元,成交额2...
中证A500ETF华安(159... 8月22日,截止午间收盘,中证A500ETF华安(159359)涨1.15%,报1.139元,成交额...
科创AIETF(588790)... 8月22日,截止午间收盘,科创AIETF(588790)涨4.83%,报0.760元,成交额6.98...
创业板50ETF嘉实(1593... 8月22日,截止午间收盘,创业板50ETF嘉实(159373)涨2.61%,报1.296元,成交额1...
港股异动丨航空股大幅走低 中国... 港股航空股大幅下跌,其中,中国国航跌近7%表现最弱,中国东方航空跌近5%,中国南方航空跌超3%,美兰...
电网设备ETF(159326)... 8月22日,截止午间收盘,电网设备ETF(159326)跌0.25%,报1.198元,成交额409....
红利ETF国企(530880)... 8月22日,截止午间收盘,红利ETF国企(530880)跌0.67%,报1.034元,成交额29.0...