首先先上代碼
#include <utility>
#include <iostream>
using std::cout;
using std::endl;
template
<typename T>
void process(const T& t)
{
cout << "lr" << endl;
}
template
<typename T>
void process(T&& t)
{
cout << "rv" << endl;
}
template
<typename T>
void test(T&& t)
{
process(std::forward<T>(t));
}
int main()
{
int i = 1;
test(i);
getchar();
return 0;
}
以我的(不正確)的理解來看,我預(yù)想中的代碼行為應(yīng)該是打印出“l(fā)r”,即調(diào)用了接受const T&參數(shù)的process。test(i)接受一個左值實參,根據(jù)右值引用的推導(dǎo)規(guī)則,T應(yīng)被推導(dǎo)為int&, 經(jīng)過std::forward后,根據(jù)引用折疊的規(guī)則,t為int&,匹配到 void process(const T& t)這個版本。但最后實驗結(jié)果打印出的是"rv"。在此想請教一下這段代碼到底是如何執(zhí)行的。
既然你了解了引用折疊, 我相信你也應(yīng)該知道了forward就是一簡單的static_cast<T&&>t
.
此函數(shù)void process(T&& t)
是有問題的, 它依舊是一個universal reference/forwarding reference
, 只有void process(int&& t)
這樣明確是右值引用你才能稱作rv
, 對吧. 所以先改下函數(shù)并簡化代碼:
template <typename T> void process(const T& t) { cout << "const T&" << endl; }
template <typename T> void process(T&& t) { cout << "T&&" << endl; }
void test(...) { process(...) ;}
因為forward只是一個轉(zhuǎn)發(fā)(從上面的實現(xiàn)配合引用折疊也是很好理解的), 并且能保留原有的ref-qualifier和const-qualifier, 所以被稱作完美轉(zhuǎn)發(fā), 因此你可以把test里面的process繼續(xù)簡化掉:
int non_const_a = 1;
int const const_a = 1;
template <typename T> void process(const T& t) { cout << "const T&" << endl; }
template <typename T> void process(T&& t) { cout << "T&&" << endl; }
test(1); // T&&
test(non_const_a); // T&&
test(const_a); // const T&
有沒有發(fā)現(xiàn)什么? 整個過程其實就是簡化成左值, 右值, 加上const-qualifier對process的函數(shù)重載決議了.
T&&
還是const T&
都和標(biāo)題中的forward, 右值引用沒什么關(guān)系了
這下應(yīng)該明白了吧? 只有const的左值才會匹配const T&
, 其他都會匹配T&&
. 很明了的一個重載決議.
繼續(xù), 可能OP會想, 如果我這么重載呢?
template <typename T> void process(const T& t) { cout << "const T&" << endl; }
template <typename T> void process(const T&& t) { cout << "T&&" << endl; }
都有const
呀, 此時該怎么辦呢? 編譯器是不是就gg了?
簡單的說, 此時const T&&
不再是人見人愛花見花開的, forwarding reference
, 因為有了const-qualifier, 所以退化成了rvalue-reference
了. g(i)妄想傳個左值進去不是作么. 比如這樣的例子:
void f(int&& a) {}
int main()
{
int a = 1;
f(a);
}
prog.cc:5:12: error: cannot bind rvalue reference of type 'int&&' to lvalue of type 'int'
f(a);
有了以上鋪墊, OP是不是能想出之前提的問題:
#include <utility>
#include <iostream>
using std::cout;
using std::endl;
template
<typename T>
void process(const T& t)
{
cout << "const T&" << endl;
}
template
<typename T>
void process(const T&& t)
{
cout << "const T&&" << endl;
}
template
<typename T>
void test(T&& t)
{
process(std::forward<T>(t));
}
int main()
{
int a = 1;
const int const_a = 1;
test(1);
test(a);
test(const_a);
}
const T&&
const T&
const T&
可見只有右值1
匹配了const T&&
, 畢竟人家只能匹配右值嘛, 也是應(yīng)該的.
北大青鳥APTECH成立于1999年。依托北京大學(xué)優(yōu)質(zhì)雄厚的教育資源和背景,秉承“教育改變生活”的發(fā)展理念,致力于培養(yǎng)中國IT技能型緊缺人才,是大數(shù)據(jù)專業(yè)的國家
北大青鳥中博軟件學(xué)院創(chuàng)立于2003年,作為華東區(qū)著名互聯(lián)網(wǎng)學(xué)院和江蘇省首批服務(wù)外包人才培訓(xùn)基地,中博成功培育了近30000名軟件工程師走向高薪崗位,合作企業(yè)超4
中公教育集團創(chuàng)建于1999年,經(jīng)過二十年潛心發(fā)展,已由一家北大畢業(yè)生自主創(chuàng)業(yè)的信息技術(shù)與教育服務(wù)機構(gòu),發(fā)展為教育服務(wù)業(yè)的綜合性企業(yè)集團,成為集合面授教學(xué)培訓(xùn)、網(wǎng)
達內(nèi)教育集團成立于2002年,是一家由留學(xué)海歸創(chuàng)辦的高端職業(yè)教育培訓(xùn)機構(gòu),是中國一站式人才培養(yǎng)平臺、一站式人才輸送平臺。2014年4月3日在美國成功上市,融資1
浪潮集團項目經(jīng)理。精通Java與.NET 技術(shù), 熟練的跨平臺面向?qū)ο箝_發(fā)經(jīng)驗,技術(shù)功底深厚。 授課風(fēng)格 授課風(fēng)格清新自然、條理清晰、主次分明、重點難點突出、引人入勝。
曾工作于聯(lián)想擔(dān)任系統(tǒng)開發(fā)工程師,曾在博彥科技股份有限公司擔(dān)任項目經(jīng)理從事移動互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍懿科技有限責(zé)任公司從事總經(jīng)理職務(wù)負責(zé)iOS教學(xué)及管理工作。
精通HTML5和CSS3;Javascript及主流js庫,具有快速界面開發(fā)的能力,對瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁制作和網(wǎng)頁游戲開發(fā)。
具有10 年的Java 企業(yè)應(yīng)用開發(fā)經(jīng)驗。曾經(jīng)歷任德國Software AG 技術(shù)顧問,美國Dachieve 系統(tǒng)架構(gòu)師,美國AngelEngineers Inc. 系統(tǒng)架構(gòu)師。