網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
function
目前,我們的知識(shí)深度已知的可調(diào)用對(duì)象類型有:
- 函數(shù)指針
- 仿函數(shù) / 函數(shù)對(duì)象
- lambda表達(dá)式
現(xiàn)在我們有一個(gè)函數(shù)模板
template<class F, class T> T useF(F f, T x) { static int count = 0; cout << "count:" << &count << endl; return f(x); }
對(duì)于函數(shù)模板,編譯器會(huì)根據(jù)實(shí)參,按照模板定義出一份特定的函數(shù)。
函數(shù)內(nèi)部的靜態(tài)成員變量是屬于函數(shù)的,無(wú)論調(diào)用多少次該函數(shù),都只會(huì)定義出一個(gè)。
記住上面這兩個(gè)知識(shí)點(diǎn),現(xiàn)在增加一個(gè)函數(shù)和仿函數(shù),用來(lái)測(cè)試useF
函數(shù)模板
int Sub(int num) { return (num - 2); } struct Func { int operator()(int num) { return (num - 3); } }; int main() { // 函數(shù)名 cout << useF(Sub, 4) << endl; // 函數(shù)對(duì)象 cout << useF(Func(), 4) << endl; // lambda表達(dá)式 cout << useF([](int d)->int{ return (d - 4); }, 11.11) << endl; return 0; }
解釋運(yùn)行結(jié)果:我們?cè)诤瘮?shù)模板內(nèi)部實(shí)現(xiàn)打印靜態(tài)成員變量,發(fā)現(xiàn)三次打印的cout地址不一樣。然而靜態(tài)成員變量是屬于函數(shù)的,一個(gè)函數(shù)的靜態(tài)成員變量無(wú)論調(diào)用多少次都只有一份。這說(shuō)明是三個(gè)不同的函數(shù)調(diào)用。
以lambda表達(dá)式為例,一個(gè)lambda表達(dá)式語(yǔ)句就生成一個(gè)自定義類型(仿函數(shù)),那么多次調(diào)用會(huì)根據(jù)模板產(chǎn)生非常多的函數(shù)。
int main() { // 函數(shù)名 cout << useF(Sub, 4) << endl; // 函數(shù)對(duì)象 cout << useF(Func(), 4) << endl; // lamber表達(dá)式 cout << useF([](int d)->int{ return (d - 4); }, 11.11) << endl; cout << useF([](int d)->int { return (d - 4); }, 11.11) << endl; return 0; }
其他可調(diào)用對(duì)象的類型也是很多的,許多的函數(shù)指針,許多的仿函數(shù)類,許多的lambda表達(dá)式……。類型太豐富了!對(duì)于一個(gè)模板而言,類型不同,就會(huì)對(duì)應(yīng)定義出一份。模板的效率也降低了太多。
C++11提供了包裝器,包裝器可以將可調(diào)用對(duì)象統(tǒng)一包裝成一個(gè)類型。function就是一個(gè)包裝器,也可稱為適配器
function
#include <functional> template <class Ret, class... Args> class function<Ret(Args...)>;
Ret(Args…):第一個(gè)模板參數(shù)類型(參數(shù)包)
測(cè)試:
template<class F, class T> T useF(F f, T x) { static int count = 0; //cout << "count:" << ++count << endl; cout << "count:" << &count << endl; return f(x); } int main() { function<int(int)> f1 = [](int d)->int { return (d - 4); }; function<int(int)> f2 = [](int d)->int { return (d - 4); }; function<int(int)> f3 = [](int d)->int { return (d - 4); }; // lamber表達(dá)式 cout << useF(f1, 5) << endl; cout << useF(f2, 5) << endl; cout << useF(f3, 5) << endl; return 0; }
上面調(diào)用的都是同一個(gè)函數(shù)。
一個(gè)lambda表達(dá)式語(yǔ)句會(huì)生成一個(gè)類,上面有三個(gè)lambda表達(dá)式語(yǔ)句,生成三個(gè)類。使用function包裝器將這些可調(diào)用對(duì)象包裝成了一個(gè)類型,模板也就只需要定義出一份特定的,極大地提升了模板的效率。
【普通函數(shù)指針】
包裝用法:function<Ret(Args...)> 對(duì)象名 = 函數(shù)指針
//例如 int Sub(int x, int y) { return x - y; } int main(void) { //包裝函數(shù)指針 function<int(int, int)> f1 = Sub; cout << f1(12, 8) << endl; return 0; }
【仿函數(shù)】
包裝用法:function<Ret(Args……) 對(duì)象名 = 仿函數(shù)類()
class Sub { public: int operator()(int x, int y) { return (x - y); } }; int main(void) { function<int(int, int)> f2 = Sub(); return 0; }
【靜態(tài)類成員函數(shù)指針】
包裝方法:function<Ret(Args……) 對(duì)象名 = &類域::函數(shù)指針
& 可以不加,不影響結(jié)果,但是加上要更優(yōu)一些。
class Sub { public: static int SubStatic(int x, int y) { return (x - y); } }; int main(void) { function<int(int, int)> f2 = &Sub::SubStatic; cout << f2(10, 3) << endl; function<int(int, int)> f3 = Sub::SubStatic; cout << f2(10, 3) << endl; return 0; }
【非靜態(tài)類成員函數(shù)指針】
包裝方法:function<Ret(類域名, Args……) 對(duì)象名 = &類域::函數(shù)指針
class Sub { public: int SubMember(int x, int y) { return (x - y); } }; int main(void) { function<int(Sub, int, int)> f4 = Sub::SubMember; cout << f4(Sub(), 3, 1) << endl; return 0; }
非靜態(tài)類成員函數(shù)指針包裝后,使用包裝后的對(duì)象進(jìn)行調(diào)用,第一個(gè)參數(shù),必須是類名。
【lambda表達(dá)式】
包裝方法:function<Ret(Args……) 對(duì)象名 = lambda表達(dá)式
int main(void) { function<double(double, double)> f5 = [](double x, double y)mutable->double {return x - y; }; cout << f5(2.23, 1.11) << endl; return 0; }
bind
bind也是一個(gè)包裝器。
作用一:調(diào)整參數(shù)的順序
普通函數(shù)指針的包裝方法:function<Ret(Args...)> 對(duì)象名 = bind(函數(shù)指針,newArgs……)
int Sub(int x, int y) { return x - y; } int main(void) { //function、bind包裝函數(shù)指針 function<int(int, int)> f1 = Sub; //將第一個(gè)參數(shù)和第二個(gè)參數(shù)交換 function<int(int, int)> f2 = bind(Sub, placeholders::_2, placeholders::_1); cout << f1(12, 8) << endl; cout << f2(12, 8) << endl; return 0; }
placeholders::_n,表示當(dāng)前function類中參數(shù)包的第n個(gè)參數(shù)。
希望怎么調(diào)整參數(shù)的順序,就在調(diào)用bind函數(shù)時(shí),傳遞什么樣的參數(shù)順序。bind函數(shù)傳參時(shí),使用placeholders::_n。
作用二、指定某個(gè)參數(shù)的值
#include <iostream> #include <functional> using namespace std; int Sub(int x, int y) { return x - y; } int main(void) { //function、bind包裝函數(shù)指針 function<int(int, int)> f2 = bind(Sub, 10, placeholders::_2); cout << f2(12, 8) << endl; return 0; }
使用function和bind包裝過(guò)后,并且指定了某個(gè)參數(shù)的值。function實(shí)例化的時(shí)候可以省略掉指定了值的參數(shù)的參數(shù)類型。省略掉后要注意維護(hù)bind函數(shù)內(nèi)的參數(shù)包。
int Sub(int x, int y) { return x - y; } int main(void) { //function、bind包裝函數(shù)指針 function<int(int)> f2 = bind(Sub, 10, placeholders::_1); cout << f2(8) << endl; return 0; }
可得出結(jié)論,bind的間接作用:調(diào)整參數(shù)的個(gè)數(shù)。
原文鏈接:https://blog.csdn.net/qq_56870066/article/details/126473985
相關(guān)推薦
- 2023-03-30 一鍵移除ButterKnife并替換為ViewBinding的舊項(xiàng)目拯救_Android
- 2022-09-25 mybatis增刪改查實(shí)現(xiàn)以及模糊查詢
- 2021-12-02 C++11?constexpr使用詳解_C 語(yǔ)言
- 2022-04-16 C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之二叉鏈表創(chuàng)建二叉樹_C 語(yǔ)言
- 2022-08-28 Oracle觸發(fā)器和程序包的基本介紹_oracle
- 2022-11-10 Android使用popupWindow仿微信彈出框使用方法_Android
- 2022-07-29 Android錄音功能的實(shí)現(xiàn)以及踩坑實(shí)戰(zhàn)記錄_Android
- 2022-07-14 python倒序for循環(huán)實(shí)例_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支