0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

關(guān)于C++仿函數(shù)

工程師 ? 來源:dnbc66 ? 作者:dnbc66 ? 2020-10-23 15:28 ? 次閱讀

【導(dǎo)讀】:在我們?nèi)粘>幋a中會發(fā)現(xiàn)有些功能代碼,會不斷的在不同的成員函數(shù)中用到,但是又不好將這些代碼獨(dú)立成一個(gè)成員函數(shù)。解決辦法之一就是寫一個(gè)公共的函數(shù),不過函數(shù)用到的一些變量,就可能會成為全局變量。再說為了復(fù)用這么一段代碼,就要單立出一個(gè)函數(shù),也不是很好維護(hù)。此時(shí)就可以用到仿函數(shù)了。

以下是正文

引入仿函數(shù)(functor)原因

先考慮一個(gè)簡單的例子:假設(shè)有一個(gè)vector《string》,你的任務(wù)是統(tǒng)計(jì)長度小于5的string的個(gè)數(shù),如果使用count_if函數(shù)的話,你的代碼可能長成這樣:

bool LengthIsLessThanFive(const string& str){ return str.length() 《 5; }int res=count_if(vec.begin(), vec.end(), LengthIsLessThanFive);

其中count_if函數(shù)的第三個(gè)參數(shù)是一個(gè)函數(shù)指針,返回一個(gè)bool類型的值。一般的,如果需要將特定的閾值長度也傳入的話,我們可能將函數(shù)寫成這樣:

bool LenthIsLessThan(const string& str, int len) { return str.length() 《 len;}

這個(gè)函數(shù)看起來比前面一個(gè)版本更具有一般性,但是他不能滿足count_if函數(shù)的參數(shù)要求:count_if要求的是unary function(僅帶有一個(gè)參數(shù))作為它的最后一個(gè)參數(shù)。所以問題來了,怎么樣找到以上兩個(gè)函數(shù)的一個(gè)折中的解決方案呢?

這個(gè)問題其實(shí)可以歸結(jié)于一個(gè)data flow的問題,要設(shè)計(jì)這樣一個(gè)函數(shù),使其能夠access這個(gè)特定的length值,回顧我們已有的知識,有三種解決方案可以考慮:

(1)函數(shù)的局部變量:

局部變量不能在函數(shù)調(diào)用中傳遞,而且caller無法訪問。

(2)函數(shù)的參數(shù):

這種方法我們已經(jīng)討論過了,多個(gè)參數(shù)不適用于count_if函數(shù)。

(3)全局變量:

我們可以將長度閾值設(shè)置成一個(gè)全局變量,代碼可能像這樣:

int maxLength;bool LengthIsLessThan(const string& str) { return str.length() 《 maxLength;}int res=count_if(vec.begiin(), vec.end(), LengthIsLessThan);

這段代碼看似很不錯(cuò),實(shí)則不符合規(guī)范,更重要的是,它不優(yōu)雅。原因有以下幾點(diǎn)要考慮:

(1)容易出錯(cuò):

為什么這么說呢,我們必須先初始化maxLength的值,才能繼續(xù)接下來的工作,如果我們忘了,則可能無法得到正確答案。此外,變量maxLength和函數(shù)LengthIsLessThan之間是沒有必然聯(lián)系的,編譯器無法確定在調(diào)用該函數(shù)前是否將變量初始化,給碼農(nóng)平添負(fù)擔(dān)。

(2)沒有可擴(kuò)展性:

如果我們每遇到一個(gè)類似的問題就新建一個(gè)全局變量,尤其是多人合作寫代碼時(shí),很容易引起命名空間污染(namespace polution)的問題;當(dāng)范圍域內(nèi)有多個(gè)變量時(shí),我們用到的可能不是我們想要的那個(gè)。

(3)全局變量的問題:

每當(dāng)新建一個(gè)全局變量,即使是為了coding的便利,我們也要知道我們應(yīng)該盡可能的少使用全局變量,因?yàn)樗腸ost很高;而且可能暗示你這里有一些待解決的優(yōu)化方案。

仿函數(shù)(functor)介紹

說了這么多,還是要回到我們原始的那個(gè)問題,有什么解決方案呢?答案當(dāng)然就是這篇blog的正題部分:仿函數(shù)。

我們的初衷是想設(shè)計(jì)一個(gè)unary function,使其能做binary function的工作,這看起來并不容易,但是仿函數(shù)能解決這個(gè)問題。

先來看仿函數(shù)的通俗定義:仿函數(shù)(functor)又稱為函數(shù)對象(function object)是一個(gè)能行使函數(shù)功能的類。仿函數(shù)的語法幾乎和我們普通的函數(shù)調(diào)用一樣,不過作為仿函數(shù)的類,都必須重載operator()運(yùn)算符,舉個(gè)例子:

class Func{ public: void operator() (const string& str) const { cout《《str《《endl; }};Func myFunc;myFunc(“helloworld!”);

》》》helloworld!

仿函數(shù)其實(shí)是上述解決方案中的第四種方案:成員變量。成員函數(shù)可以很自然的訪問成員變量:

class StringAppend{ public: explicit StringAppend(const string& str) : ss(str){} void operator() (const string& str) const{ cout《《str《《‘ ’《《ss《《endl; } private: const string ss; }; StringAppend myFunc(“is world”); myFunc(“hello”);

》》》hellois world

我相信這個(gè)例子能讓你體會到一點(diǎn)點(diǎn)仿函數(shù)的作用了;它既能像普通函數(shù)一樣傳入給定數(shù)量的參數(shù),還能存儲或者處理更多我們需要的有用信息。

讓我們回到count_if的問題中去,是不是覺得問題變得豁然開朗了?

class ShorterThan { public: explicit ShorterThan(int maxLength) : length(maxLength) {} bool operator() (const string& str) const { return str.length() 《 length; } private: const int length;};//直接調(diào)用即可count_if(myVector.begin(), myVector.end(), ShorterThan(length));

這里需要注意的是,不要糾結(jié)于語法問題:ShorterThan(length)似乎并沒有調(diào)用operator()函數(shù)?其實(shí)它調(diào)用了,創(chuàng)建了一個(gè)臨時(shí)對象。你也可以自己加一些輸出語句看一看。

這篇博文就先記到這里了,仿函數(shù)也在STL中大量涉及到,不徹底弄懂仿函數(shù)的問題看到STL源碼就會一頭包。后續(xù)可能再分享一些關(guān)于functor的資料和個(gè)人學(xué)習(xí)心得。

-END-

來源 | dnbc66

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4237

    瀏覽量

    61969
  • C++
    C++
    +關(guān)注

    關(guān)注

    21

    文章

    2085

    瀏覽量

    73302
收藏 人收藏

    評論

    相關(guān)推薦

    C++語言基礎(chǔ)知識

    電子發(fā)燒友網(wǎng)站提供《C++語言基礎(chǔ)知識.pdf》資料免費(fèi)下載
    發(fā)表于 07-19 10:58 ?6次下載

    C++中實(shí)現(xiàn)類似instanceof的方法

    函數(shù),可實(shí)際上C++中沒有。但是別著急,其實(shí)C++中有兩種簡單的方法可以實(shí)現(xiàn)類似Java中的instanceof的功能。 在 C++ 中,確定對象的類型是編程中實(shí)際需求,使開發(fā)人員
    的頭像 發(fā)表于 07-18 10:16 ?355次閱讀
    <b class='flag-5'>C++</b>中實(shí)現(xiàn)類似instanceof的方法

    FX2 CY7C68013A如何在C++環(huán)境中使用LoadEEPROM函數(shù)?

    我使用的是 FX2 CY7C68013A 芯片。 我知道 CyUSB.NET 庫中有我需要的 LoadEEPROM 函數(shù)。 請問如何在 C++ 環(huán)境而不是 C#/CLR 環(huán)境中使用該
    發(fā)表于 05-31 06:59

    使用 MISRA C++:2023? 避免基于范圍的 for 循環(huán)中的錯(cuò)誤

    在前兩篇博客中,我們?向您介紹了新的 MISRA C++ 標(biāo)準(zhǔn)?和?C++ 的歷史?。在這篇博客中,我們將仔細(xì)研究以 C++ 中?for?循環(huán)為中心的特定規(guī)則。
    的頭像 發(fā)表于 03-28 13:53 ?519次閱讀
    使用 MISRA <b class='flag-5'>C++</b>:2023? 避免基于范圍的 for 循環(huán)中的錯(cuò)誤

    c語言,c++,java,python區(qū)別

    C語言、C++、Java和Python是四種常見的編程語言,各有優(yōu)點(diǎn)和特點(diǎn)。 C語言: C語言是一種面向過程的編程語言。它具有底層的特性,能夠?qū)τ?jì)算機(jī)硬件進(jìn)行直接操作。
    的頭像 發(fā)表于 02-05 14:11 ?1366次閱讀

    C++簡史:C++是如何開始的

    MISRA C++:2023,MISRA? C++ 標(biāo)準(zhǔn)的下一個(gè)版本,來了!為了幫助您做好準(zhǔn)備,我們介紹了 Perforce 首席技術(shù)支持工程師 Frank van den Beuken 博士撰寫
    的頭像 發(fā)表于 01-11 09:00 ?427次閱讀
    <b class='flag-5'>C++</b>簡史:<b class='flag-5'>C++</b>是如何開始的

    基于C/C++面向?qū)ο蟮姆绞椒庋bsocket通信類

    在掌握了基于 TCP 的套接字通信流程之后,為了方便使用,提高編碼效率,可以對通信操作進(jìn)行封裝,本著有淺入深的原則,先基于 C 語言進(jìn)行面向過程的函數(shù)封裝,然后再基于 C++ 進(jìn)行面向?qū)ο蟮念惙庋b。
    的頭像 發(fā)表于 12-26 09:57 ?1140次閱讀

    C語言和C++中那些不同的地方

    C語言雖說經(jīng)常和C++在一起被大家提起,但可千萬不要以為它們是一個(gè)東西。現(xiàn)在我們常用的C語言是C89標(biāo)準(zhǔn),C++
    的頭像 發(fā)表于 12-07 14:29 ?773次閱讀
    <b class='flag-5'>C</b>語言和<b class='flag-5'>C++</b>中那些不同的地方

    如何選擇創(chuàng)建c語言和c++

    選擇創(chuàng)建 C 語言和 C++ 都需要綜合考慮多個(gè)因素。在決定使用哪種語言之前,我們需要對這兩種語言的特點(diǎn)、優(yōu)缺點(diǎn)、適用場景、學(xué)習(xí)成本等進(jìn)行全面的了解和對比。下面是關(guān)于選擇創(chuàng)建 C 語言
    的頭像 發(fā)表于 11-27 15:58 ?458次閱讀

    c++怎么開始編程

    C++是一種高級的、通用的編程語言,用于開發(fā)各種類型的應(yīng)用程序。它是從C語言演變而來,也是一種靜態(tài)類型語言,可以在不同的平臺上進(jìn)行開發(fā)。C++具有高度的靈活性和性能,并且廣泛應(yīng)用于游戲開發(fā)、桌面
    的頭像 發(fā)表于 11-27 15:56 ?737次閱讀

    探索C++的編程習(xí)慣與編程要點(diǎn)

    C++的類可以分為帶指針數(shù)據(jù)成員與不帶指針數(shù)據(jù)成員兩類,complex就屬于不帶指針成員的類。而這里要說的字符串類String,一般的實(shí)現(xiàn)會帶有一個(gè)char *指針。帶指針數(shù)據(jù)成員的類,需要自己實(shí)現(xiàn)class三大件:拷貝構(gòu)造函數(shù)、拷貝賦值
    的頭像 發(fā)表于 11-14 09:25 ?330次閱讀
    探索<b class='flag-5'>C++</b>的編程習(xí)慣與編程要點(diǎn)

    什么是C++函數(shù)? 應(yīng)該怎么定義? 用途是什么?

    什么是C++函數(shù)? 應(yīng)該怎么定義? 主要用途是什么?
    發(fā)表于 11-08 06:58

    如何綁架C語言標(biāo)準(zhǔn)庫函數(shù)?

    C語言動態(tài)內(nèi)存可以使用 malloc 標(biāo)準(zhǔn)庫函數(shù),而 C++ 會使用 new 關(guān)鍵字申請,裸機(jī)情況下一般沒有問題,但是在 RTOS 情況下,就需要考慮資源保護(hù)的情況
    發(fā)表于 11-01 09:50 ?277次閱讀

    C++之父新作帶你勾勒現(xiàn)代C++地圖

    為了幫助大家解決這些痛點(diǎn)問題,讓大家領(lǐng)略現(xiàn)代C++之美,掌握其中的精髓,更好地使用C++C++之父Bjarne Stroustrup坐不住了,他親自操刀寫就了這本《C++之旅》!
    的頭像 發(fā)表于 10-30 16:35 ?698次閱讀
    <b class='flag-5'>C++</b>之父新作帶你勾勒現(xiàn)代<b class='flag-5'>C++</b>地圖

    C++程序設(shè)計(jì)精簡版

    章 指針第 7 章 自定義數(shù)據(jù)類型 第 3 篇 基于對象的程序設(shè)計(jì)第 8 章 類和對象第 9 章 關(guān)于類和對象的進(jìn)一步討論第 10 章 運(yùn)算符重載 第 4 篇 面向?qū)ο蟮某绦蛟O(shè)計(jì)第 11 章 繼承與派生第 12 章 多態(tài)性與虛函數(shù)第 13 章 輸入輸出流第 14 章
    發(fā)表于 10-09 07:26