探索仿函数(Functor):C++中的灵活函数对象

news2025/1/16 21:17:08

文章目录

  • 一、仿函数定义及使用
  • 二、仿函数与函数指针的区别
  • 三、仿函数与算法的关系
  • 四、仿函数的实践用例

在C++编程中,我们经常需要对数据进行排序、筛选或者其他操作。为了实现这些功能,C++标准库提供了许多通用的算法和容器,而其中一个重要的概念就是 仿函数(Functor)。仿函数是一种行为类似函数的对象,它可以被当做函数使用,作为算法的参数传递,或者在容器中使用。本文将深入探讨仿函数的概念、用法以及在实际开发中的应用,希望能够帮助读者更好地理解和应用仿函数这一重要的编程概念。

一、仿函数定义及使用

仿函数(Functor) 是一种行为类似函数的对象,它可以被用作函数并接受参数。在C++中,仿函数通常是重载了函数调用运算符operator()的类对象。通过重载operator(),仿函数可以像函数一样被调用,并且可以保存状态信息。

按照操作数划分:假定某个类有一个重载的operator(),而且重载的operator()要求获取一个参数,我们就将这个类称为一元仿函数(unary functor);如果重载的operator()要求获取两个参数,就将这个类称为二元仿函数(binary functor)

按照功能划分:可分为算数运算(Arithmetic)关系运算(Rational)、**逻辑运算(Logical)**三大类。

一个简单的例子是比较仿函数,它可以用来对数据进行排序。例如,可以定义一个比较仿函数来比较两个整数的大小,并将其传递给STL中的sort函数来进行排序。

以下是一个简单的比较仿函数的示例:

// 定义一个比较仿函数
template<class T>
struct LessThan {
    bool operator()(T a, T b) const { return a < b;}
};
int main() {
    LessThan<int> lessThan; // 创建比较仿函数对象
    std::cout << lessThan(3, 5) << std::endl; // 使用仿函数对象进行比较
    std::cout << LessThan<int>()(3, 5) << std::endl; // 使用仿函数对象进行比较
    return 0;
}

在这个例子中,LessThan是一个仿函数,重载了operator()来进行比较。在main函数中,我们创建了LessThan的对象lessThan,并使用它来比较两个整数的大小。要将某种操作当做算法的参数,唯一的办法就是先将该操作(可能用于数条以上的指令)设计为一个函数,再将函数指针当作一个算法的一个参数,或是将该操作设计为一个所谓的仿函数(从语言层次来看就是一个class),再以该仿函数产生一个对象,并以此对象作为算法参数的一部分。

//一元仿函数,
struct SetKeyOfT{
    const K& operator()(const K& key){ return key; }
};
struct MapKeyOfT{
    const K& operator()(const pair<K, V>& kv) { return kv.first; }
};

这两个仿函数(也称为函数对象)分别用于提取不同类型数据的键(key)。仿函数的主要目的是提供一个可调用对象,通常用于算法中作为自定义的比较或操作函数。

仿函数SetKeyOfT是设计用来从单独的键(类型为K)中提取键本身。它重载了operator(),使其能够接受一个类型为K的常量引用作为参数,并返回该键的常量引用。

仿函数MapKeyOfT用于从pair<K, V>中提取键(first成员)。在C++标准库中,pair是一个模板类,通常用于表示键值对,例如在mapunordered_map等容器中。这个仿函数通过重载operator(),使其能够接受一个pair<K, V>类型的常量引用作为参数,并返回该pair的键(first成员)的常量引用。

这在实际应用中可能看起来有些多余,但是在我们模仿实现一些数据结构时可以起到作用(如仿写set 和 map)。 通过使用仿函数,我们可以将函数对象作为参数传递给其他函数,或者将其存储在容器中,从而实现更灵活的编程和功能组合。


二、仿函数与函数指针的区别

函数指针也可以当作参数传递给算法当参数。但是函数指针不能满足 STL对抽象性的要求,即无法和STL其他组件搭配。

在STL(标准模板库)中,仿函数(也称为函数对象)与算法之间存在着密切的关系。STL算法通常接受仿函数作为参数,以便能够自定义算法的行为。这种灵活性使得STL算法能够适用于各种不同的场景,而不仅仅是预定义的操作。

仿函数(functor)和函数指针在编程中虽然都有其独特的应用场景,但它们在实现方式、使用灵活性和抽象层次等方面存在显著的区别。

首先,仿函数是一个类的实例,通过重载operator()操作符,使得这个类的对象可以像函数一样被调用。这使得仿函数在行为上表现得像一个函数,但实际上它拥有类的所有特性,如数据成员和成员函数。因此,仿函数可以封装更复杂的状态和行为,并且可以在类定义中提供更多的灵活性和控制。

相比之下,函数指针是一个指向函数的指针变量。它本身是一个指针,指向的是函数的入口地址。函数指针主要用于在运行时动态地调用不同的函数,或者将函数作为参数传递给其他函数。然而,函数指针的使用受到一定的限制,因为它只能指向已定义的函数,而不能封装更复杂的状态或行为。

在抽象层次上,仿函数提供了比函数指针更高层次的抽象。仿函数可以看作是函数指针的泛化,它不仅能够像函数指针一样动态地调用不同的函数,还能够封装更多的状态和行为。这使得仿函数在使用STL算法等需要高度抽象和灵活性的场合中更为适用。


三、仿函数与算法的关系

首先仿函数在STL中的作用是极大的。

仿函数在STL中的主要作用是提供一种可以像函数一样调用的对象。它们通常通过重载operator()来定义自己的行为,从而可以在算法中作为参数传递,以决定算法如何操作元素。

算法通常与仿函数进行结合

STL算法是高度通用化的,它们通过接受仿函数作为参数来适应不同的操作需求。例如,std::sort算法可以对容器进行排序,但它并不直接定义如何比较元素。相反,它接受一个仿函数作为参数,该仿函数定义了如何比较元素。这样,你可以为不同的数据类型或排序需求提供不同的比较逻辑。

在这里插入图片描述

同样,std::transform算法可以对容器中的元素进行转换,它接受一个仿函数来定义转换的逻辑。你可以提供一个仿函数来执行任何你想要的转换操作。下面是一个使用仿函数进行排序的示例:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
// 定义一个仿函数,用于比较两个整数的大小
struct CompareInts {
    bool operator()(const int& a, const int& b) const{ return a < b; }
    // 升序比较
};
int main() {
    vector<int> vec = {5, 2, 8, 1, 9};
    // 使用sort算法和自定义的仿函数进行排序
    sort(vec.begin(), vec.end(), CompareInts());
    return 0;
}

在这个例子中,我们定义了一个名为CompareInts的仿函数,它重载了operator()来定义如何比较两个整数。然后,我们将这个仿函数作为第三个参数传递给sort算法,以便按照升序对整数进行排序。

⚠️我们如果要使用STL内建的仿函数,都必须含 <functional>头文件。


四、仿函数的实践用例

我们拿leetoce中的题目来做案例:692. 前K个高频单词 - 力扣(LeetCode)。题目如下:

给定一个单词列表 words 和一个整数 k ,返回前 k 个出现次数最多的单词。

返回的答案应该按单词出现频率由高到低排序。如果不同的单词有相同出现频率, 按字典顺序 排序。

示例 1:

输入: words = ["i", "love", "leetcode", "i", "love", "coding"], k = 2
输出: ["i", "love"]
解析: "i" 和 "love" 为出现次数最多的两个单词,均为2次。
    注意,按字母顺序 "i" 在 "love" 之前。

示例 2:

输入: ["the", "day", "is", "sunny", "the", "the", "the", "sunny", "is", "is"], k = 4
输出: ["the", "is", "sunny", "day"]
解析: "the", "is", "sunny" 和 "day" 是出现次数最多的四个单词,
    出现次数依次为 4, 3, 2 和 1 次。

注意:

  • 1 <= words.length <= 500
  • 1 <= words[i] <= 10
  • words[i] 由小写英文字母组成。
  • k 的取值范围是 [1, 不同 words[i] 的数量]
class Solution {
public:
    class Com{
    public:
        bool operator()(const pair<string,int> &kv1, const pair<string,int> &kv2){
            return kv1.second > kv2.second || 
            (kv1.second == kv2.second && kv1.first < kv2.first) ;
        }
    };
    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int > mp;
        for(auto &e : words)
            mp[e]++;

        vector<pair<string,int>> ans(mp.begin(),mp.end()); 
        sort(ans.begin(),ans.end(),Com());
        auto it = ans.begin();

        vector<string> ret;
        while(k--){
            ret.push_back(it->first);
            it++;
        }
        return ret;
    }
};

Com是一个嵌套的仿函数,它重载了operator()以提供自定义的比较逻辑。这个仿函数用于对pair<string, int>类型的元素进行比较,其中string代表单词,int代表该单词的出现频率。

整体解题思路如下:

  1. 统计频率topKFrequent函数首先遍历输入的字符串数组words,并使用map数据结构mp统计每个单词的出现频率。
  2. 创建向量:接着,它将map中的元素(键值对)复制到一个vector<pair<string, int>>类型的向量ans中。
  3. 排序:然后,它使用sort函数对ans进行排序。排序时使用了之前定义的仿函数Com作为比较函数,因此排序结果会按照单词频率的降序和字典序的升序进行排列。
  4. 提取结果:最后,程序使用一个迭代器it遍历排序后的ans,并将前k个单词(即频率最高的k个单词)添加到结果向量ret中。
  5. 返回结果:函数返回包含前k个最频繁单词的ret向量。

本题通过使用仿函数实现了自定义的比较逻辑,这种比较逻辑确保了在排序后,出现频率高的单词会排在前面,如果频率相同,则字典序小的单词排在前面。使得我们可以按照特定的顺序对单词进行排序,并最终提取出出现频率最高的前k个单词。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1519066.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux--基本知识入门

一.几个基本知识 终端: CtrlAltT 或者桌面/文件夹右键,打开终端切换为管理员: sudo su 退出:exit查看内核版本号: uname -a内核版本号含义: 5 代表主版本号;13代表次版本号;0代表修订版本号;30代表修订版本的第几次微调;数字越大表示内核越新. 二.目录…

git bash 命令行反应慢、卡顿(定位出根本原因)

参考该博主&#xff1a; https://blog.csdn.net/weixin_50212044/article/details/131575987?utm_mediumdistribute.pc_relevant.none-task-blog-2defaultbaidujs_baidulandingword~default-0-131575987-blog-130024908.235v43pc_blog_bottom_relevance_base4&spm1001.210…

26.网络游戏逆向分析与漏洞攻防-网络通信数据包分析工具-实现生成日志文件的功能

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 如果看不懂、不知道现在做的什么&#xff0c;那就跟着做完看效果 内容参考于&#xff1a;易道云信息技术研究院VIP课 上一个内容&#xff1a;25.利用全新的通…

【北京大学】徐高《金融经济学二十五讲》

一、经济的任务 经济的任务之一是确保有效地分配稀缺资源&#xff0c;这是经济学中的一个核心问题。资源是有限的&#xff0c;而需求是无限的&#xff0c;因此经济系统需要通过合理的机制来分配资源以满足社会的需求。以下是关于经济分配资源的几个方面&#xff1a; 1. 资源配…

Matlab进阶绘图第45期—蝴蝶气泡图

蝴蝶气泡图是一种特殊的柱泡图/气泡柱状图。 蝴蝶图一般由左右两个水平柱状图组合而成&#xff0c;其形如蝴蝶展翅&#xff0c;可以很直观地展示两种数据直接的差异。 而蝴蝶气泡图则是在两个水平柱状图每根柱子外侧额外添加大小不同的气泡&#xff0c;用于表示另外一个数据变…

web学习笔记(三十三)

目录 1.严格模式 1.1严格模式的概念&#xff1a; 1.2严格模式在语义上更改的地方&#xff1a; 1.3如何开启严格模式 1.4严格模式应用上的变化 2.原型链 1.严格模式 1.1严格模式的概念&#xff1a; 严格模式有点像es5向es6过渡而产生的一种模式&#xff0c;因为es6的语法…

密码保护小贴士:如何应对常见的网络钓鱼攻击?

网络钓鱼攻击是一种常见的网络欺诈手段&#xff0c;针对个人隐私和财产安全构成威胁。以下是一些密码保护的小贴士&#xff0c;帮助您应对常见的网络钓鱼攻击&#xff1a; 1.谨慎点击链接&#xff1a;收到来历不明的邮件、短信或社交媒体消息时&#xff0c;不要轻易点击其中的…

案例分析篇09:Web架构设计相关20个考点(7~11)(2024年软考高级系统架构设计师冲刺知识点总结)

专栏系列文章推荐: 2024高级系统架构设计师备考资料(高频考点&真题&经验)https://blog.csdn.net/seeker1994/category_12593400.html 【历年案例分析真题考点汇总】与【专栏文章案例分析高频考点目录】(2024年软考高级系统架构设计师冲刺知识点总结-案例分析篇-…

深度学习 精选笔记(11)深度学习计算相关:GPU、参数、读写、块

学习参考&#xff1a; 动手学深度学习2.0Deep-Learning-with-TensorFlow-bookpytorchlightning ①如有冒犯、请联系侵删。 ②已写完的笔记文章会不定时一直修订修改(删、改、增)&#xff0c;以达到集多方教程的精华于一文的目的。 ③非常推荐上面&#xff08;学习参考&#x…

PS学习 - 抠图-通道

抠出蝴蝶 1.通道抠图 套索工具 这里需要圈住你要的&#xff0c;注意尽量小点 ctrl j 复制 然后去掉背景 点击通道 找到明暗对比最大的通道&#xff0c;这里我理解为颜色反差最大的那个&#xff0c;突出你要抠的东西 搜了下说是一般为蓝色 复制通道 ctrll调出色阶 通过移…

前端:pre实现JSON格式化和搜索词高亮

产品功能 最终结果如下图所示&#xff0c;将获取到的json数据格式化&#xff0c;并且在搜索时&#xff0c;将搜索的关键词高亮。 实现 HTML <!-- 搜索框 --> <el-input class"input" v-model"searchjsonValue" change"searchJSON" p…

【Linux】Centos7安装Nginx1.21.6

Nginx是一款轻量级的Web 服务器/反向代理服务器及电子邮件( IMAP/POP3)代理服务器。其特点是占有内存少&#xff0c;并发能力强&#xff0c;事实上nginx的并发能力在同类型的网页服务器中表现较好&#xff0c;中国大陆使nginx的网站有:百度、京东、新浪、网易、腾讯、淘宝等。 …

Android Gradle 编译过程中的优化

} 但是反射没有办法能很好的识别&#xff0c;所以如果代码中有使用反射需要自行处理&#xff0c;以免被删除。 shrink resource 功能 shrink code流程执行完后删除了无用的代码后&#xff0c;就能确认哪些资源文件没有使用&#xff0c;shrink resource流程就是确定哪些资源…

人工智能(AI)+、+了什么?互联网+又是什么?

引言 人工智能&#xff08;Artificial Intelligence&#xff0c;简称AI&#xff09;&#xff0c;作为一门迅猛发展的领域&#xff0c;自20世纪中叶以来一直在不断演进。它涉及计算机科学、机器学习和模式识别等多个学科&#xff0c;旨在开发可以模拟和执行人类智力任务的系统。…

Unity PS5开发 天坑篇 之 DEVKit环境部署与系统升级02

上一篇各位大神们已经收到了SONY官方免费寄送的PS5开发机与测试机&#xff0c;恭喜大家成为SONY的开发者, 本篇继续PS5开发机的部署与开发套件使用。 一, PC安装PS5 SDK与系统升级 1. PC/PS5 SDK Manager下载安装包 登录开发者账号后&#xff0c;Development->Resources&a…

bdd100k数据集格式转coco格式

最近在学习使用mmdetection&#xff0c;需要使用bdd100k数据集来训练网络&#xff0c;但是官网下载的数据集格式不是coco数据集&#xff0c;得自己转换数据集格式。 文章目录 一、bdd100k数据集下载二、bdd100k转coco脚本下载三、脚本使用报错 一、bdd100k数据集下载 数据集下…

单片机卡死的几大原因、分析、解决

阅读引言&#xff1a; 本文我想给大家分享一下我在学习过程中遇到的以及了解到的一些导致单片机运行卡死&#xff08;死锁&#xff09;的一些常见原因和解决办法&#xff0c; 请注意&#xff0c; 只是列举&#xff0c;并不是全部&#xff0c; 因为导致单片机运行卡死的原因无穷…

三款内衣洗衣机的顶级较量:希亦、小吉、由利,谁才是性价比之王?

洗衣机在我们的生活中可谓是非常常见的了&#xff0c;几乎每家每户都具备着一台。即便是有洗衣机&#xff0c;也有不少人不会将自己我贴身衣物直接扔在洗衣机里清洗&#xff0c;而是会自己手工手洗。这跟我们传统上的观念有很大的关系&#xff0c;认为把内衣、内裤等贴身衣物放…

MongoDB实战面试指南:常见问题一网打尽

码到三十五 &#xff1a; 个人主页 心中有诗画&#xff0c;指尖舞代码&#xff0c;目光览世界&#xff0c;步履越千山&#xff0c;人间尽值得 ! MongoDB是一款流行的非关系型数据库&#xff0c;以其高效、可扩展的特性受到开发者的青睐。了解MongoDB的架构、存储引擎和数据结…

【C++】手撕vector类(从会用到理解)

一、标准库中的vector类 1.1 vector类介绍 1.2 vector的常用接口 1.2.1 常用的构造函数 1.2.2 容量操作接口 &#xff08;1&#xff09;size &#xff08;2&#xff09;capacity &#xff08;3&#xff09;empty &#xff08;4&#xff09;resize &#xff08;5&#…