当前位置:首页 > > 21ic电子网
[导读]在我们日常编码中会发现有些功能代码,会不断的在不同的成员函数中用到,但是又不好将这些代码独立成一个成员函数。解决办法之一就是写一个公共的函数,不过函数用到的一些变量,就可能会成为全局变量。再说为了复用这么一段代码,就要单立出一个函数,也不是很好维护。此时就可以用到仿函数了。

【导读】:在我们日常编码中会发现有些功能代码,会不断的在不同的成员函数中用到,但是又不好将这些代码独立成一个成员函数。解决办法之一就是写一个公共的函数,不过函数用到的一些变量,就可能会成为全局变量。再说为了复用这么一段代码,就要单立出一个函数,也不是很好维护。此时就可以用到仿函数了。


以下是正文



引入仿函数(functor)原因


先考虑一个简单的例子:假设有一个vector,你的任务是统计长度小于5的string的个数,如果使用count_if函数的话,你的代码可能长成这样:

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

其中count_if函数的第三个参数是一个函数指针,返回一个bool类型的值。一般的,如果需要将特定的阈值长度也传入的话,我们可能将函数写成这样:

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

这个函数看起来比前面一个版本更具有一般性,但是他不能满足count_if函数的参数要求:count_if要求的是unary function(仅带有一个参数)作为它的最后一个参数。所以问题来了,怎么样找到以上两个函数的一个折中的解决方案呢?


这个问题其实可以归结于一个data flow的问题,要设计这样一个函数,使其能够access这个特定的length值,回顾我们已有的知识,有三种解决方案可以考虑:


(1)函数的局部变量:

局部变量不能在函数调用中传递,而且caller无法访问。


(2)函数的参数:

这种方法我们已经讨论过了,多个参数不适用于count_if函数。


(3)全局变量:

我们可以将长度阈值设置成一个全局变量,代码可能像这样:

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

这段代码看似很不错,实则不符合规范,更重要的是,它不优雅。原因有以下几点要考虑:


(1)容易出错:

为什么这么说呢,我们必须先初始化maxLength的值,才能继续接下来的工作,如果我们忘了,则可能无法得到正确答案。此外,变量maxLength和函数LengthIsLessThan之间是没有必然联系的,编译器无法确定在调用该函数前是否将变量初始化,给码农平添负担。


(2)没有可扩展性:

如果我们每遇到一个类似的问题就新建一个全局变量,尤其是多人合作写代码时,很容易引起命名空间污染(namespace polution)的问题;当范围域内有多个变量时,我们用到的可能不是我们想要的那个。


(3)全局变量的问题:

每当新建一个全局变量,即使是为了coding的便利,我们也要知道我们应该尽可能的少使用全局变量,因为它的cost很高;而且可能暗示你这里有一些待解决的优化方案。


仿函数(functor)介绍


说了这么多,还是要回到我们原始的那个问题,有什么解决方案呢?答案当然就是这篇blog的正题部分:仿函数。


我们的初衷是想设计一个unary function,使其能做binary function的工作,这看起来并不容易,但是仿函数能解决这个问题。


先来看仿函数的通俗定义:仿函数(functor)又称为函数对象(function object)是一个能行使函数功能的类。仿函数的语法几乎和我们普通的函数调用一样,不过作为仿函数的类,都必须重载operator()运算符,举个例子:

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

>>>helloworld!


仿函数其实是上述解决方案中的第四种方案:成员变量。成员函数可以很自然的访问成员变量:

class StringAppend{ public: explicit StringAppend(const string& str) : ss(str){}
void operator() (const string& str) const{ cout<' '< }
private: const string ss; };
StringAppend myFunc("is world"); myFunc("hello");

>>>hellois world


我相信这个例子能让你体会到一点点仿函数的作用了;它既能像普通函数一样传入给定数量的参数,还能存储或者处理更多我们需要的有用信息。


让我们回到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;};//直接调用即可count_if(myVector.begin(), myVector.end(), ShorterThan(length));

这里需要注意的是,不要纠结于语法问题:ShorterThan(length)似乎并没有调用operator()函数?其实它调用了,创建了一个临时对象。你也可以自己加一些输出语句看一看。


这篇博文就先记到这里了,仿函数也在STL中大量涉及到,不彻底弄懂仿函数的问题看到STL源码就会一头包。后续可能再分享一些关于functor的资料和个人学习心得。

C++仿函数你知道怎么做吗?


来源:dnbc66

链接:www.cnblogs.com/decade-dnbc66/p/5347088.html

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

21ic电子网

扫描二维码,关注更多精彩内容

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读

全球嵌入式技术领域的年度盛会2026嵌入式世界展(Embedded World 2026,简称EW26)于3月10日至12日在德国纽伦堡成功举办。作为物联网和边缘AI领域的领先企业,Silicon Labs(亦称“芯科科...

关键字: 物联网 边缘AI 嵌入式

3月10日至12日,2026年嵌入式世界展(Embedded World 2026,简称EW26)在德国纽伦堡展览中心成功举办。作为领先的边缘AI与智能音频等媒体处理技术和芯片解决方案提供商,XMOS以沉浸式演示与技术交...

关键字: 边缘AI 智能音频 嵌入式

在嵌入式系统开发中,SoC(System on Chip)的多样性始终是横亘在开发者面前的难题。以某工业物联网网关项目为例,其需同时支持NXP i.MX8M、Rockchip RK3566和Allwinner H616三...

关键字: Platform Driver模型 嵌入式

在高性能网络编程领域,事件驱动模型以其高效的I/O多路复用能力成为主流范式。不同于传统的多线程/多进程阻塞模型,事件驱动通过单一线程监听多个文件描述符的状态变化,以非阻塞方式处理I/O事件,显著减少了上下文切换开销和资源...

关键字: 事件驱动 C语言

在非易失性存储器领域,EEPROM(电可擦除可编程只读存储器)曾长期占据主流地位,广泛应用于各类电子设备的参数存储、日志记录等场景。但随着工业控制、汽车电子、医疗设备等领域对存储性能提出更高要求,FRAM(铁电随机存取存...

关键字: 存储器 可编程 嵌入式

康佳特将aReady.COM扩展至Arm架构模块,基于恩智浦i.MX 95处理器打造应用就绪的软硬件构建模块,集成操作系统、系统整合与IoT连接能力,赋能高价值应用快速落地

关键字: 处理器 IoT 嵌入式

3月12日,2026年中国家电及消费电子博览会(以下简称:AWE 2026)在上海盛大开幕。展会现场,场景化、系统化、一体化的家电解决方案成为行业焦点,消费者对家电的关注也已从基础的尺寸匹配,延伸至对“空间秩序感”和“视...

关键字: 消费电子 蒸烤箱 嵌入式

超高效NPU IP在资源受限设备中推进边缘AI,因而获得认可

关键字: 人工智能 嵌入式 NPU

上海2026年3月12日 /美通社/ -- 3月12日,2026中国家电及消费电子博览会(AWE 2026)在上海新国际博览中心和东方枢纽国际商务合作区展区正式启幕。本届展会以"AI科技•慧享未来"为...

关键字: 西门子 博世 嵌入式 洗碗机
关闭