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