C++小利器之std::bind參數(shù)綁定包裝器的使用詳解
C++ 原生支持函數(shù)輸入?yún)?shù)的默認(rèn)值,但是有些業(yè)務(wù)場(chǎng)景下對(duì)原有設(shè)定的默認(rèn)值不滿意,那么可不可以臨時(shí)改改?
請(qǐng)注意,隨意改變?cè)写a邏輯,并不符合軟件設(shè)計(jì)原則。有沒(méi)有什么方案或者特性可以做到不修改原有代碼邏輯,以實(shí)現(xiàn)新增參數(shù)默認(rèn)值(或者在調(diào)用時(shí)綁定特性參數(shù))的功能?
從 C++11 開(kāi)始,標(biāo)準(zhǔn)庫(kù)提供了 std::bind 用于綁定函數(shù) f 和調(diào)用參數(shù),返回一個(gè)新可調(diào)用函數(shù)對(duì)象 fn (也有喊它做仿函數(shù)的)。于是,在調(diào)用函數(shù)對(duì)象 fn 和輸入新參數(shù)時(shí),相當(dāng)于調(diào)用了原來(lái)的函數(shù) f 并自動(dòng)按照綁定的關(guān)系映射參數(shù),映射的方式非常靈活。這樣被綁定映射的參數(shù),實(shí)現(xiàn)原有函數(shù) f 的參數(shù)新默認(rèn)值功能,或者減少調(diào)用函數(shù) f 時(shí)輸入的參數(shù)個(gè)數(shù)等便捷需求。
簡(jiǎn)而言之,std::bind 被稱為轉(zhuǎn)發(fā)調(diào)用包裝器。
綁定時(shí),通過(guò)占位符 _1, _2, _3… ,指定在調(diào)用函數(shù)對(duì)象 fn 的輸入?yún)?shù)和傳遞給函數(shù) f 的參數(shù)之間的映射位置。
被綁定的函數(shù) f 也分種類,那么,基于不同類型函數(shù)的應(yīng)用示例是怎樣的呢?
綁定靜態(tài)或者全局函數(shù)
#include <functional> void output1(int a) { printf("%d\n", a); } void output2(int a, int b) { printf("%d %d\n", a, b); } int main(int argc, char * argv[]) { auto fn1 = std::bind(output1, std::placeholders::_1); fn1(1); auto fn2 = std::bind(output2, std::placeholders::_2, 0); fn2(3, 2); return 0; }
output:
1
2 0
占位符代表函數(shù)對(duì)象的輸入?yún)?shù),std::placeholders::_1 對(duì)應(yīng)調(diào)用函數(shù)對(duì)象 fn 時(shí)的第 1 個(gè)輸入?yún)?shù),依次類推。
細(xì)心的你可能會(huì)留意 std::bind 返回的是可調(diào)用對(duì)象,這個(gè)和同樣從 C++11 開(kāi)始引入的 Lambda 表達(dá)式概念是類似的,那么有沒(méi)有可能用 Lambda 表達(dá)式互相替換?試試改為使用 Lambda 表達(dá)式實(shí)現(xiàn)
int main(int argc, char * argv[]) { auto fn1 = [](int a) -> void { output1(a); }; fn1(1); auto fn2 = [](int a, int b) -> void { output2(b, 0); }; fn2(3, 2); return 0; }
這樣子替換后,執(zhí)行輸出結(jié)果是一樣的。
output:
1
2 0
綁定對(duì)象成員函數(shù)
如果被綁定的函數(shù)是對(duì)象成員,需要指定所屬對(duì)象,看代碼
#include <functional> class A { public: void output(int a, int b) { printf("%d %d\n", a, b); } }; int main(int argc, char * argv[]) { A obj; auto fn = std::bind(&A::output, &obj, std::placeholders::_2, std::placeholders::_1); fn(5, 3); return 0; }
output:
3 5
用 Lambda 表達(dá)式替換試試
int main(int argc, char * argv[]) { A obj; auto fn = [&obj](int a, int b) -> void { obj.output(b, a); }; fn(5, 3); return 0; }
output:
3 5
Notice:綁定的時(shí)候需要指定上下文環(huán)境(實(shí)例對(duì)象),這種場(chǎng)景已經(jīng)涉及到閉包的概念了。由于上下文環(huán)境是有生命周期的,比如上面用到的實(shí)例對(duì)象 a,如果在超出了其生命周期的情況下,引用其內(nèi)部已被釋放的資源,是會(huì)引起程序執(zhí)行異常的,這里提醒一下。
平替 Lambda
以上面兩種場(chǎng)景示例代碼的比較,在參數(shù)綁定的場(chǎng)景下,為了達(dá)到相同效果,Lambda 表達(dá)式的書寫略顯羅嗦,而 std::bind 的格式更有函數(shù)式編程的味道,可見(jiàn),某些場(chǎng)景下推薦 std::bind 替代 Lambda 表達(dá)式。
能不能平替也分場(chǎng)景,三思而行。
到此這篇關(guān)于C++小利器之std::bind參數(shù)綁定包裝器的使用詳解的文章就介紹到這了,更多相關(guān)C++ std::bind內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01緩存處理函數(shù)storageKeySuffix操作示例解析
這篇文章主要介紹了淺析緩存處理函數(shù)storageKeySuffix操作示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08C++通過(guò)ofstream和ifstream實(shí)現(xiàn)文件讀寫
本文詳細(xì)介紹了C++中通過(guò)ofstream和ifstream進(jìn)行文件的讀寫操作,包括文件打開(kāi)模式的選擇、錯(cuò)誤處理策略以及二進(jìn)制文件的特殊處理,重點(diǎn)講解了如何使用這些類進(jìn)行文本和二進(jìn)制文件的讀寫,并強(qiáng)調(diào)了正確處理文件操作中可能出現(xiàn)的錯(cuò)誤,需要的朋友可以參考下2025-03-03使用c++調(diào)用windows打印api進(jìn)行打印的示例代碼
這篇文章主要介紹了使用c++調(diào)用windows打印api進(jìn)行打印的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06C語(yǔ)言實(shí)現(xiàn)ATM機(jī)存取款系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)ATM機(jī)存取款系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11C語(yǔ)言手把手教你實(shí)現(xiàn)貪吃蛇AI(上)
這篇文章主要介紹了C語(yǔ)言手把手教你實(shí)現(xiàn)貪吃蛇AI,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01