C++11 std::function和std::bind 的使用示例詳解
概述
C++11中的
std::function
和std::bind
是函數(shù)對象的重要組成部分,它們可以用于將函數(shù)和參數(shù)綁定在一起,形成一個可調(diào)用的對象。std::function
可以存儲任意可調(diào)用對象,包括函數(shù)指針、函數(shù)對象、lambda表達(dá)式等,而std::bind
則可以將函數(shù)和參數(shù)綁定在一起,形成一個新的可調(diào)用對象。它們的使用可以大大簡化代碼,提高代碼的可讀性和可維護(hù)性。
可調(diào)用對象
C++中有如下幾種可調(diào)用對象,
函數(shù)、函數(shù)指針、lambda表達(dá)式、bind對象、函數(shù)對象
。
其中,lambda表達(dá)式和bind對象是C++11標(biāo)準(zhǔn)中提出的(bind機(jī)制并不是新標(biāo)準(zhǔn)中首次提出,而是對舊版本中bind1st和bind2st的合并)。
std::function
std::function是一個可調(diào)用對象包裝器,是一個類模板,可以容納除了類成員函數(shù)指針之外的所有可調(diào)用對象,它可以用統(tǒng)一的方式處理函數(shù)、函數(shù)對象、函數(shù)指針,并允許保存和延遲它們的執(zhí)行。
使用std::function可以實現(xiàn)回調(diào)函數(shù)、事件處理等功能。
std::function函數(shù)原型
#include <functional> template<class R, class... Args> class function<R(Args...)>; //其中,R 表示返回值類型,Args... 表示參數(shù)類型列表。 //例如,function<int(float, double)> 表示一個返回值為 int,接受一個 float 和一個 double 類型參數(shù)的函數(shù)對象。
std::function的主要作用
- 對C++中各種可調(diào)用實體(普通函數(shù)、Lambda表達(dá)式、函數(shù)指針、以及其它函數(shù)對象等)的封裝,形成一個新的可調(diào)用的std::function對象,簡化調(diào)用;
- 對C++中現(xiàn)有的可調(diào)用實體的一種類型安全的包裹(如:函數(shù)指針這類可調(diào)用實體,是類型不安全的)。
- 將函數(shù)作為參數(shù)傳遞給其他函數(shù);
- 將函數(shù)作為返回值返回;
- 將函數(shù)對象作為參數(shù)傳遞給其他函數(shù);
- 將函數(shù)對象作為返回值返回。
//例如,定義一個返回值為int,參數(shù)為兩個int的函數(shù)對象: std::function<int(int, int)>func; //可以將一個函數(shù)指針或lambda表達(dá)式賦值給函數(shù)對象: int add(int a, int b) { return a +b; } func = add; // 函數(shù)指針賦值 func = [](int a, int b) { return a + b; };// lambda表達(dá)式賦值 //調(diào)用函數(shù)對象可以使用operator(),例如: int result = func(1, 2); // 調(diào)用add函數(shù),返回3
std::function的優(yōu)缺點
- 優(yōu)點:
可以方便地實現(xiàn)回調(diào)函數(shù)、事件處理等功能,同時也可以用于實現(xiàn)函數(shù)對象的封裝和傳遞。
- 缺點:
它的使用會帶來一定的性能開銷,因為它需要在運行時進(jìn)行類型檢查和動態(tài)分配內(nèi)存。
此外,如果使用不當(dāng),也容易引起內(nèi)存泄漏和對象生命周期管理的問題。
std::bind
std::function是一個可調(diào)用對象包裝器,是一個類模板,可以容納除了類成員函數(shù)指針之外的所有可調(diào)用對象,它可以用統(tǒng)一的方式處理函數(shù)、函數(shù)對象、函數(shù)指針,并允許保存和延遲它們的執(zhí)行。
std::bind函數(shù)原型
template<class F, class... Args> /unspecified/ bind(F&& f, Args&&... args); //其中,F(xiàn)是要綁定的函數(shù)對象,Args是要綁定的參數(shù)。返回值是一個新的可調(diào)用對象,可以直接調(diào)用或者存儲起來后再調(diào)用。
std::bind的主要作用
- 將可調(diào)用對象和其參數(shù)綁定成一個仿函數(shù);
- 只綁定部分參數(shù),減少可調(diào)用對象傳入的參數(shù)。
- 綁定函數(shù)對象的參數(shù),生成一個新的可調(diào)用對象,可以方便地將函數(shù)對象作為參數(shù)傳遞給其它函數(shù)。
- 可以將成員函數(shù)綁定到對象上,生成一個新的可調(diào)用對象,方便地調(diào)用成員函數(shù)。
- 可以將成員函數(shù)綁定到對象指針上,生成一個新的可調(diào)用對象,方便地調(diào)用成員函數(shù)。
- 可以將成員函數(shù)綁定到對象引用上,生成一個新的可調(diào)用對象,方便地調(diào)用成員函數(shù)。
- 可以將函數(shù)對象綁定到函數(shù)指針上,生成一個新的可調(diào)用對象,方便地調(diào)用函數(shù)對象。
- 可以將函數(shù)對象綁定到函數(shù)引用上,生成一個新的可調(diào)用對象,方便地調(diào)用函數(shù)對象。
- 可以將函數(shù)對象綁定到std::function對象上,生成一個新的可調(diào)用對象,方便地調(diào)用函數(shù)對象。
例如,我們有一個函數(shù)對象:void foo(int a, int b, int c) { std::cout << a << " " << b << " " << c << std::endl; }
我們可以使用std::bind將它綁定到一些參數(shù)上:auto f = std::bind(foo, 1, 2, 3);
這里,f是一個新的可調(diào)用對象,它綁定了foo函數(shù)和參數(shù)1、2、3。我們可以像調(diào)用原始函數(shù)對象一樣調(diào)用它:f(); // 輸出:1 2 3
我們也可以只綁定部分參數(shù):auto g = std::bind(foo, 1, std::placeholders::_1, 3);
這里,std::placeholders::_1表示占位符,它表示在調(diào)用g時,第二個參數(shù)會被傳遞給foo函數(shù)。我們可以這樣調(diào)用g:g(2); // 輸出:1 2 3
這就是std::bind的基本用法。它可以方便地將函數(shù)對象和參數(shù)綁定在一起,生成一個新的可調(diào)用對象。
std::bind的優(yōu)缺點
優(yōu)點:
可以方便地實現(xiàn)函數(shù)對象的復(fù)用和參數(shù)的延遲綁定,從而提高代碼的可讀性和可維護(hù)性。
缺點:
可能會導(dǎo)致代碼的復(fù)雜性增加,特別是當(dāng)參數(shù)較多時,需要謹(jǐn)慎使用。
代碼示例
#pragma once #include <iostream> #include <functional> class A { public: bool TESTA(int, char*, int) { /* implementation */ } }; class B { public: bool TESTB(std::function<bool(int, char*, int)> func) { /* implementation */ } }; int main() { A objA; B objB; auto lambda = [](int a, char* b, int c) { /* implementation */ }; objB.TESTB(lambda); objB.TESTB(std::bind(&A::TESTA, &objA, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3)); return 0; }
總結(jié)
- 預(yù)綁定的參數(shù)是以值傳遞的形式,不預(yù)綁定的參數(shù)要用std::placeholders(占位符)的形式占位,從_1開始,依次遞增,是以引用傳遞的形式;
- std::placeholders表示新的可調(diào)用對象的第幾個參數(shù),而且與原函數(shù)的該占位符所在位置的進(jìn)行匹配;
- bind綁定類成員函數(shù)時,第一個參數(shù)表示對象的成員函數(shù)的指針,第二個參數(shù)表示對象的地址,這是因為對象的成員函數(shù)需要有this指針。并且編譯器不會將對象的成員函數(shù)隱式轉(zhuǎn)換成函數(shù)指針,需要通過&手動轉(zhuǎn)換;
- std::bind的返回值是可調(diào)用實體,可以直接賦給std::function。
到此這篇關(guān)于C++11 std::function和std::bind 的使用的文章就介紹到這了,更多相關(guān)C++11 std::function和std::bind內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ 學(xué)習(xí)之旅 Windows程序內(nèi)部運行原理
學(xué)習(xí)C++與.net不同的是,一定要搞清楚Windows程序內(nèi)部運行原理,因為他所涉及大多數(shù)是操作系統(tǒng)的調(diào)用,而.net畢竟是在.netFrameWork上唱戲2012-11-11C語言數(shù)據(jù)結(jié)構(gòu)之堆排序詳解
堆是計算機(jī)科學(xué)中一類特殊的數(shù)據(jù)結(jié)構(gòu)的統(tǒng)稱,通常是一個可以被看做一棵完全二叉樹的數(shù)組對象。而堆排序是利用堆這種數(shù)據(jù)結(jié)構(gòu)所設(shè)計的一種排序算法。本文將通過圖片詳細(xì)介紹堆排序,需要的可以參考一下2022-03-03C語言數(shù)據(jù)結(jié)構(gòu)之二叉鏈表創(chuàng)建二叉樹
這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)之?二叉鏈表創(chuàng)建二叉樹,下文我們?yōu)榱烁奖愕氖褂枚鏄浣Y(jié)構(gòu)體,可以使用?typedef?對結(jié)構(gòu)體進(jìn)行命名,具體內(nèi)容需要的小伙伴可以參考一下2022-02-02kernel劫持modprobe?path內(nèi)容詳解
這篇文章主要為大家介紹了kernel劫持modprobe?path的內(nèi)容詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05