C++編程之?std::forward使用例子
std::forward 是一個(gè) C++11 中的模板函數(shù),其主要作用是在模板函數(shù)或模板類中,將一個(gè)參數(shù)以“原樣”(forward)的方式轉(zhuǎn)發(fā)給另一個(gè)函數(shù)。通常情況下,該函數(shù)被用于實(shí)現(xiàn)完美轉(zhuǎn)發(fā)(perfect forwarding)。
完美轉(zhuǎn)發(fā)是指,一個(gè)函數(shù)或類模板可以將其參數(shù)原封不動(dòng)地轉(zhuǎn)發(fā)給另一個(gè)函數(shù)或類模板,同時(shí)保持被轉(zhuǎn)發(fā)參數(shù)的左右值特性(lvalue 或 rvalue)。它在實(shí)現(xiàn)泛型編程時(shí)非常有用,因?yàn)樗梢员苊庵貜?fù)編寫代碼,同時(shí)提高代碼的可復(fù)用性。
在 C++ 中,函數(shù)參數(shù)可以是左值引用(lvalue reference)或右值引用(rvalue reference)。對于一個(gè)模板函數(shù)或類模板,當(dāng)傳遞一個(gè)參數(shù)時(shí),如果該參數(shù)是左值,那么傳遞的就是一個(gè)左值引用;如果該參數(shù)是右值,那么傳遞的就是一個(gè)右值引用。
通常情況下,在將參數(shù)轉(zhuǎn)發(fā)給其他函數(shù)時(shí),我們需要保留原始參數(shù)的左右值特性。這就是 std::forward 函數(shù)的作用,它可以將一個(gè)參數(shù)的左右值特性原封不動(dòng)地轉(zhuǎn)發(fā)給其他函數(shù)。
下面是一個(gè)使用 std::forward 的例子:
#include <iostream> #include <utility> void func(int& x) { std::cout << "lvalue reference: " << x << std::endl; } void func(int&& x) { std::cout << "rvalue reference: " << x << std::endl; } template<typename T> void wrapper(T&& arg) { func(std::forward<T>(arg)); } int main() { int x = 42; wrapper(x); // lvalue reference: 42 wrapper(1); // rvalue reference: 1 return 0; }
在上面的例子中,我們定義了兩個(gè)函數(shù) func,一個(gè)接受左值引用,另一個(gè)接受右值引用。然后我們定義了一個(gè)模板函數(shù) wrapper,它的參數(shù)是一個(gè)完美轉(zhuǎn)發(fā)引用(perfect forwarding reference) T&&。在 wrapper 函數(shù)中,我們使用 std::forward 函數(shù)將參數(shù) arg 轉(zhuǎn)發(fā)給 func 函數(shù)。通過使用 std::forward,我們可以確保 func 函數(shù)接收到的參數(shù)的左右值特性與原始參數(shù)保持一致。
- 當(dāng)向wrapper里面?zhèn)魅離的時(shí)候,wrapper推導(dǎo)認(rèn)為 T是一個(gè)左值引用int &,通過引用折疊原則(看萬能引用文章)int && + & = int &,相當(dāng)于wrapper(int& arg),同時(shí)我們知道了T推導(dǎo)為int&,那么在向func傳遞的時(shí)候,就是func(std::forward<int&> (arg)) ,那么func會(huì)以左值引用的形式 func(int& x) 調(diào)用arg。
- 當(dāng)向wrapper里面?zhèn)魅?的時(shí)候,wrapper推導(dǎo)認(rèn)為T是一個(gè)右值引用int&& ,通過引用折疊原則,int && + && =int&& ,相當(dāng)于wrapper(int&& arg),同時(shí)我們知道了T推導(dǎo)為int&&,那么在向func傳遞的時(shí)候,就是func(std::forward<int&&>(arg)),那么func會(huì)以左值引用的形式func(int&& x)調(diào)用arg。
簡單來說,當(dāng)傳遞給 wrapper 函數(shù)的參數(shù)是右值時(shí),T 會(huì)被推導(dǎo)為右值引用類型 int&&,此時(shí) std::forward(arg) 的返回值類型為 int&&,將會(huì)調(diào)用 func(int&&)。當(dāng)傳遞給 wrapper 函數(shù)的參數(shù)是左值時(shí),T 會(huì)被推導(dǎo)為左值引用類型 int&,此時(shí) std::forward(arg) 的返回值類型為 int&,將會(huì)調(diào)用 func(int&)。
當(dāng)我們把std::forward去掉的話,那么當(dāng)傳入一個(gè)具名變量參數(shù)時(shí),func會(huì)認(rèn)為這個(gè)值就是是一個(gè)左值。當(dāng)傳入一個(gè)臨時(shí)變量(不具名變量)參數(shù)的時(shí)候,func會(huì)認(rèn)為這個(gè)值就是一個(gè)右值。
#include <iostream> #include <utility> void func(int&& x) { std::cout << "rvalue reference: " << x << std::endl; } void func(int& x) { std::cout << "lvalue reference: " << x << std::endl; } template<typename T> void wrapper(T&& arg) { func(arg); func(1); } int main() { int x = 42; wrapper(x); // lvalue reference: 42 wrapper(10); // rvalue reference: 1 return 0; }
輸出:
lvalue reference: 42
rvalue reference: 1
lvalue reference: 10
rvalue reference: 1
這樣的結(jié)果就是不能對arg推導(dǎo)出的類型完美轉(zhuǎn)發(fā)到其他函數(shù)中,顯然不符合本意。
到此這篇關(guān)于C++編程之 std::forward的文章就介紹到這了,更多相關(guān)C++ std::forward內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言修煉之路一朝函數(shù)思習(xí)得?模塊思維世間生下篇
函數(shù)是一組一起執(zhí)行一個(gè)任務(wù)的語句。每個(gè)?C?程序都至少有一個(gè)函數(shù),即主函數(shù)?main()?,所有簡單的程序都可以定義其他額外的函數(shù)2022-03-03C++?Cartographer源碼中關(guān)于傳感器的數(shù)據(jù)傳遞實(shí)現(xiàn)
這篇文章主要介紹了C++?Cartographer源碼中關(guān)于傳感器的數(shù)據(jù)傳遞實(shí)現(xiàn),前面已經(jīng)談到了Cartographer中添加軌跡的方法和傳感器的數(shù)據(jù)流動(dòng)走向。發(fā)現(xiàn)在此調(diào)用了LaunchSubscribers這個(gè)函數(shù)來訂閱相關(guān)傳感器數(shù)據(jù)2023-03-03C++實(shí)現(xiàn)拼圖游戲代碼(graphics圖形庫)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)拼圖游戲代碼,帶有g(shù)raphics圖形庫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05Qt實(shí)現(xiàn)驗(yàn)證碼相關(guān)功能的代碼示例
驗(yàn)證碼的原理基于人類視覺和計(jì)算機(jī)視覺的差異性,通過給用戶顯示一些難以被機(jī)器識(shí)別的圖形或文字,讓用戶進(jìn)行人機(jī)交互,確認(rèn)自己的身份,這樣可以有效保護(hù)網(wǎng)站安全,所以本給大家介紹了Qt實(shí)現(xiàn)驗(yàn)證碼相關(guān)功能的代碼示例,感興趣的朋友可以參考下2024-01-01C++ 動(dòng)態(tài)創(chuàng)建按鈕及 按鈕的消息響應(yīng)
這篇文章主要介紹了C++ 動(dòng)態(tài)創(chuàng)建按鈕及 按鈕的消息響應(yīng)的相關(guān)資料,需要的朋友可以參考下2015-06-06