首先先上代碼
#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)接受一個(gè)左值實(shí)參,根據(jù)右值引用的推導(dǎo)規(guī)則,T應(yīng)被推導(dǎo)為int&, 經(jīng)過std::forward后,根據(jù)引用折疊的規(guī)則,t為int&,匹配到 void process(const T& t)這個(gè)版本。但最后實(shí)驗(yàn)結(jié)果打印出的是"rv"。在此想請(qǐng)教一下這段代碼到底是如何執(zhí)行的。
既然你了解了引用折疊, 我相信你也應(yīng)該知道了forward就是一簡(jiǎn)單的static_cast<T&&>t.
此函數(shù)void process(T&& t)是有問題的, 它依舊是一個(gè)universal reference/forwarding reference , 只有void process(int&& t)這樣明確是右值引用你才能稱作rv, 對(duì)吧. 所以先改下函數(shù)并簡(jiǎn)化代碼:
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(...) ;}
因?yàn)閒orward只是一個(gè)轉(zhuǎn)發(fā)(從上面的實(shí)現(xiàn)配合引用折疊也是很好理解的), 并且能保留原有的ref-qualifier和const-qualifier, 所以被稱作完美轉(zhuǎn)發(fā), 因此你可以把test里面的process繼續(xù)簡(jiǎn)化掉:
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)什么? 整個(gè)過程其實(shí)就是簡(jiǎn)化成左值, 右值, 加上const-qualifier對(duì)process的函數(shù)重載決議了.
T&&還是const T&都和標(biāo)題中的forward, 右值引用沒什么關(guān)系了
這下應(yīng)該明白了吧? 只有const的左值才會(huì)匹配const T&, 其他都會(huì)匹配T&&. 很明了的一個(gè)重載決議.
繼續(xù), 可能OP會(huì)想, 如果我這么重載呢?
template <typename T> void process(const T& t) { cout << "const T&" << endl; }
template <typename T> void process(const T&& t) { cout << "T&&" << endl; }
都有const呀, 此時(shí)該怎么辦呢? 編譯器是不是就gg了?
簡(jiǎn)單的說, 此時(shí)const T&&不再是人見人愛花見花開的, forwarding reference, 因?yàn)橛辛薱onst-qualifier, 所以退化成了rvalue-reference了. g(i)妄想傳個(gè)左值進(jìn)去不是作么. 比如這樣的例子:
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)中國(guó)IT技能型緊缺人才,是大數(shù)據(jù)專業(yè)的國(guó)家
達(dá)內(nèi)教育集團(tuán)成立于2002年,是一家由留學(xué)海歸創(chuàng)辦的高端職業(yè)教育培訓(xùn)機(jī)構(gòu),是中國(guó)一站式人才培養(yǎng)平臺(tái)、一站式人才輸送平臺(tái)。2014年4月3日在美國(guó)成功上市,融資1
北大課工場(chǎng)是北京大學(xué)校辦產(chǎn)業(yè)為響應(yīng)國(guó)家深化產(chǎn)教融合/校企合作的政策,積極推進(jìn)“中國(guó)制造2025”,實(shí)現(xiàn)中華民族偉大復(fù)興的升級(jí)產(chǎn)業(yè)鏈。利用北京大學(xué)優(yōu)質(zhì)教育資源及背
博為峰,中國(guó)職業(yè)人才培訓(xùn)領(lǐng)域的先行者
曾工作于聯(lián)想擔(dān)任系統(tǒng)開發(fā)工程師,曾在博彥科技股份有限公司擔(dān)任項(xiàng)目經(jīng)理從事移動(dòng)互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍(lán)懿科技有限責(zé)任公司從事總經(jīng)理職務(wù)負(fù)責(zé)iOS教學(xué)及管理工作。
浪潮集團(tuán)項(xiàng)目經(jīng)理。精通Java與.NET 技術(shù), 熟練的跨平臺(tái)面向?qū)ο箝_發(fā)經(jīng)驗(yàn),技術(shù)功底深厚。 授課風(fēng)格 授課風(fēng)格清新自然、條理清晰、主次分明、重點(diǎn)難點(diǎn)突出、引人入勝。
精通HTML5和CSS3;Javascript及主流js庫(kù),具有快速界面開發(fā)的能力,對(duì)瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁(yè)制作和網(wǎng)頁(yè)游戲開發(fā)。
具有10 年的Java 企業(yè)應(yīng)用開發(fā)經(jīng)驗(yàn)。曾經(jīng)歷任德國(guó)Software AG 技術(shù)顧問,美國(guó)Dachieve 系統(tǒng)架構(gòu)師,美國(guó)AngelEngineers Inc. 系統(tǒng)架構(gòu)師。