当前位置: 首页 > article >正文

深入理解仿函数(Functors):从概念到实践

文章目录

  • 1. 什么是仿函数?
  • 2. 仿函数与普通函数的区别
  • 3. 标准库中的仿函数
  • 4. 仿函数的优势
    • 4.1 状态保持
    • 4.2 可定制性
    • 4.3 性能优势
  • 5. 现代C++中的仿函数
    • 5.1 Lambda表达式
    • 5.2 通用仿函数
  • 6. 仿函数的高级应用(使用C++2020标准库及以上版本)
    • 6.1 函数组合
    • 6.2 惰性求值
  • 7. 总结

1. 什么是仿函数?

仿函数(Functor),也称为函数对象(Function Object),是C++中一种特殊的对象,它可以像函数一样被调用。本质上,仿函数是一个类或结构体,它重载了函数调用运算符operator()

仿函数是STL中的重要概念之一,可以用于 实现自定义的比较、排序、映射 等操作,使代码更具灵活性和可重用性。

template<class T>
class Less 
{
public:bool operator()(const T& x, const T& y) {return x < y;}
};

2. 仿函数与普通函数的区别

特性普通函数仿函数
状态保持可以保持状态
内联优化可能更易被编译器内联
多台支持有限支持完整面向对象特性
作为模版参数仅函数指针可直接作为模版参数

3. 标准库中的仿函数

C++标准库提供了许多有用的仿函数,主要位于 “functional” 头文件中:

  • 算术运算:plus, minus, multiplies, divides, modulus, negate
  • 比较运算:equal_to, not_equal_to, greater, less, greater_equal, less_equal
  • 逻辑运算:logical_and, logical_or, logical_not
  • 位运算:bit_and, bit_or, bit_xor

这里我们用std::plus()来创建一个加法仿函数,并将其存储在std::function对象中,然后可以像函数一样去调用它。

#include <iostream>
#include <functional>int main() 
{std::function<int(int, int)> myFunc = std::plus<int>(); // 使用标准库的仿函数int result = myFunc(5, 3); // 调用仿函数std::cout << "Result: " << result << std::endl;return 0;
}

4. 仿函数的优势

4.1 状态保持

仿函数可以包含成员变量,因此可以在调用之间保持状态:

class Counter {int count = 0;
public:void operator()(int x) {std::cout << "Element #" << ++count << ": " << x << "\n";}
};int main() {std::vector<int> v = {10, 20, 30};std::for_each(v.begin(), v.end(), Counter());
}

在这里插入图片描述

4.2 可定制性

通过模板和继承,仿函数可以实现高度可定制的行为:

在这个示例中,我们创建了两个不同的仿函数,一个用于加法(MyAdditionFunc),一个用于减法( MySubtractionFunc ),它们可以根据需要进行切换。

class MyAdditionFunc 
{
public:int operator()(int a, int b) {return a + b;}
};class MySubtractionFunc 
{
public:int operator()(int a, int b) {return a - b;}
};int main() 
{MyAdditionFunc add;MySubtractionFunc subtract;int result1 = add(5, 3);        // 8int result2 = subtract(5, 3);   // 2std::cout << "Result of addition: " << result1 << std::endl;std::cout << "Result of subtraction: " << result2 << std::endl;return 0;
}

4.3 性能优势

在某些情况下,使用仿函数可能会导致性能开销,因为它们引入了额外的函数调用。与普通函数相比,仿函数可能会有一些微小的性能损失。但在 绝大多数情况下,这种性能损失非常小,通常可以忽略不计。在性能要求非常高的特定应用中,可以使用内联函数或其他优化手段来减小性能损失。

5. 现代C++中的仿函数

5.1 Lambda表达式

C++11引入的 Lambda 表达式本质上是匿名仿函数的语法糖:

auto add = [](int a, int b) { return a + b; };
std::cout << add(3, 4);  // 输出7

5.2 通用仿函数

使用模板和 auto 参数使仿函数更通用:

struct GenericAdd {template <typename T, typename U>auto operator()(T t, U u) const {return t + u;}
};int main() {GenericAdd add;std::cout << add(3, 4.5);  // 输出7.5
}

6. 仿函数的高级应用(使用C++2020标准库及以上版本)

6.1 函数组合

仿函数可以组合形成更复杂的行为:

template <typename F, typename G>
class Compose {F f;G g;public:// 构造函数:存储 f 和 gCompose(F f, G g): f(f), g(g){}// operator() 允许对象像函数一样被调用template <typename X>auto operator()(X x) const {return f(g(x)); // 先计算 g(x),再计算 f(g(x))}
};int main() {auto square = [](int x) { return x * x; };auto increment = [](int x) { return x + 1; };// Compose(f, g) 表示 f(g(x)),即先执行 g,再执行 fauto square_then_increment = Compose(increment, square); // 先平方,再加 1std::cout << square_then_increment(3); // 输出 10 (3² + 1)// 如果想先 increment 再 square,可以交换顺序auto increment_then_square = Compose(square, increment); // 先加 1,再平方std::cout << "\n" << increment_then_square(3); // 输出 16 ((3 + 1)²)
}

在这里插入图片描述

6.2 惰性求值

仿函数可以实现惰性求值模式:

class LazyValue {// function 可以将任何类型的可调用元素 (例如函数和函数对象 )包装到可复制对象中的类,并且其类型仅取决于其调用签名(而不取决于可调用元素类型本身)。std::function<int()> generator;// optional 管理一个可选 的容纳值,既可以存在也可以不存在的值。mutable std::optional<int> cache;
public:LazyValue(std::function<int()> g) : generator(g) {}operator int() const {if (!cache) {cache = generator();}return *cache;}
};int expensive_computation() {// 模拟耗时计算// sleep_for 当前线程的执行将停止,直到从现在开始至少经过 rel_time 为止。其他线程继续执行。std::this_thread::sleep_for(std::chrono::seconds(1));return 42;
}int main() {// 延迟42毫秒再运行程序LazyValue lazy(expensive_computation);// 计算尚未发生std::cout << "Value: " << lazy;  // 此时进行计算cout << endl;std::cout << "Again: " << lazy;  // 使用缓存值cout << endl;
}

在这里插入图片描述

7. 总结

仿函数是C++中强大而灵活的工具,它们:

  • 比普通函数更强大(可以保持状态)
  • 比函数指针更安全(类型安全)
  • 比虚函数更高效(静态多态)
  • 与现代C++特性(lambda、模板)完美结合

掌握仿函数将显著提升你的C++编程能力,特别是在泛型编程和标准库使用方面。

相关文章:

深入理解仿函数(Functors):从概念到实践

文章目录 1. 什么是仿函数&#xff1f;2. 仿函数与普通函数的区别3. 标准库中的仿函数4. 仿函数的优势4.1 状态保持4.2 可定制性4.3 性能优势 5. 现代C中的仿函数5.1 Lambda表达式5.2 通用仿函数 6. 仿函数的高级应用&#xff08;使用C2020标准库及以上版本&#xff09;6.1 函数…...

InternLM 论文分类微调实践(XTuner 版)

1.环境安装 我创建开发机选择镜像为Cuda12.2-conda&#xff0c;选择GPU为100%A100的资源配置 Conda 管理环境 conda create -n xtuner_101 python3.10 -y conda activate xtuner_101 pip install torch2.4.0cu121 torchvision torchaudio --extra-index-url https://downloa…...

《Python星球日记》 第88天:ChatGPT 与 LangChain

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 目录 一、ChatGPT 简介1. 如何与ChatGPT交互?2. Prompt Engineering 的技巧二、LangChain框架1. 什么是LangChain?2. LangChain的核心组件3. 如何使…...

PC:使用WinSCP密钥文件连接sftp服务器

1. 打开winscp工具&#xff0c;点击“标签页”->“新标签页” 2. 点击“高级"->“高级” 3. 点击"验证"->“选择密钥文件” 选择ppk文件&#xff0c;如果没有ppk文件选择pem文件&#xff0c;会自动生成ppk文件 点击确定 4. 输入要连接到的sftp服务器的…...

1688正式出海,1688跨境寻源通接口接入,守卫的是国内工厂资源

在1688平台的跨境招商直播中&#xff0c;许多想要进入跨境市场的初学者商家纷纷提问&#xff1a;货通全球的入口在哪里&#xff1f;小白商家应该如何操作&#xff1f;商品为何上传失败&#xff1f; 从表面上看&#xff0c;这似乎是1688平台在拓展海外市场的一次积极“进攻”。…...

力扣303 区域和检索 - 数组不可变

文章目录 题目介绍题解 题目介绍 题解 不用管第一个null&#xff0c;从第二个开始看就可以 法一&#xff1a;暴力解法 class NumArray {private int[] nums;public NumArray(int[] nums) {this.nums nums;}public int sumRange(int left, int right) {int res 0;for (int i…...

Spring的后置处理器是干什么用的?扩展点又是什么?

Spring 的后置处理器和扩展点是其框架设计的核心机制&#xff0c;它们为开发者提供了灵活的扩展能力&#xff0c;允许在 Bean 的生命周期和容器初始化过程中注入自定义逻辑。 1. 后置处理器&#xff08;Post Processors&#xff09; 后置处理器是 Spring 中用于干预 Bean 生命…...

[ linux-系统 ] 进程地址空间

验证地址空间 父子进程的变量值不同但是地址相同&#xff0c;说明该地址绝对不是物理地址 我们叫这种地址为虚拟地址/线性地址 分析与结论 上述实验表明&#xff0c;父子进程的变量地址相同但内容不同&#xff0c;说明地址为虚拟地址&#xff0c;且父子进程有各自独立的物理…...

文件名是 ‪E:\20250512_191204.mp4, EV软件录屏,未保存直接关机损坏, 如何修复?

去github上下载untrunc 工具就能修复 https://github.com/anthwlock/untrunc/releases 如果访问不了 本机的 hosts文件设置 140.82.112.3 github.com 199.232.69.194 github.global.ssl.fastly.net 就能访问了 实在不行&#xff0c;从这里下载&#xff0c;传上去了 https://do…...

Java常见API文档(下)

格式化的时间形式的常用模式对应关系如下&#xff1a; 空参构造创造simdateformate对象&#xff0c;默认格式 练习.按照指定格式展示 package kl002;import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date;public class Date3 {publi…...

DRIVEGPT4: 通过大语言模型实现可解释的端到端自动驾驶

《DriveGPT4: Interpretable End-to-End Autonomous Driving via Large Language Model》 2024年10月发表&#xff0c;来自香港大学、浙江大学、华为和悉尼大学。 多模态大型语言模型&#xff08;MLLM&#xff09;已成为研究界关注的一个突出领域&#xff0c;因为它们擅长处理…...

知识图谱(KG)与大语言模型(LLM)

知识图谱&#xff08;KG&#xff09;以其结构化的知识表示和推理能力&#xff0c;为大语言模型&#xff08;LLM&#xff09;的“幻觉”、知识更新滞后和可解释性不足等问题提供了有力的解决方案。反过来&#xff0c;LLM的强大文本理解和生成能力也为KG的构建、补全、查询和应用…...

构建共有语料库 - Wiki 语料库

中文Wiki语料库主要指的是从中文Wikipedia&#xff08;中文维基百科&#xff09;提取的文本数据。维基百科是一个自由的、开放编辑的百科全书项目&#xff0c;覆盖了从科技、历史到文化、艺术等广泛的主题。 对于基于RAG的应用来说&#xff0c;把Wiki语料作为一个公有的语料库…...

苍穹外卖项目中的 WebSocket 实战:实现来单与催单提醒功能

&#x1f680; 苍穹外卖项目中的 WebSocket 实战&#xff1a;实现来单与催单提醒功能 在现代 Web 应用中&#xff0c;实时通信成为提升用户体验的关键技术之一。WebSocket 作为一种在单个 TCP 连接上进行全双工通信的协议&#xff0c;被广泛应用于需要实时数据交换的场景&#…...

精益数据分析(59/126):移情阶段的深度博弈——如何避开客户访谈的认知陷阱

精益数据分析&#xff08;59/126&#xff09;&#xff1a;移情阶段的深度博弈——如何避开客户访谈的认知陷阱 在创业的移情阶段&#xff0c;客户访谈是挖掘真实需求的核心手段&#xff0c;但人类认知偏差往往导致数据失真。今天&#xff0c;我们结合《精益数据分析》的方法论…...

Win10 安装单机版ES(elasticsearch),整合IK分词器和安装Kibana

一. 先查看本机windows是否安装了ES(elasticsearch)&#xff0c;检查方法如下&#xff1a; 检查进程 按 Ctrl Shift Esc 组合键打开 “任务管理器”。在 “进程” 选项卡中&#xff0c;查看是否有 elasticsearch 相关进程。如果有&#xff0c;说明系统安装了 ES。 检查端口…...

Ansible模块——主机名设置和用户/用户组管理

设置主机名 ansible.builtin.hostname: name&#xff1a;要设置的主机名 use&#xff1a;更新主机名的方式&#xff08;默认会自动选择&#xff0c;不指定的话&#xff0c;物理机一般不会有问题&#xff0c;容器可能会有问题&#xff0c;一般是让它默认选择&#xff09; syst…...

【Redis】List 列表

文章目录 初识列表常用命令lpushlpushxlrangerpushrpushxlpop & rpoplindexlinsertllen阻塞操作 —— blpop & brpop 内部编码应用场景 初识列表 列表类型&#xff0c;用于存储多个字符串。在操作和实现上&#xff0c;类似 C 的双端队列&#xff0c;支持随机访问(O(N)…...

JUC入门(四)

ReadWriteLock 代码示例&#xff1a; package com.yw.rw;import java.util.HashMap; import java.util.Map; import java.util.concurrent.locks.ReentrantReadWriteLock;public class ReadWriteDemo {public static void main(String[] args) {MyCache myCache new MyCache…...

【HarmonyOS 5】鸿蒙mPaaS详解

【HarmonyOS 5】鸿蒙mPaaS详解 一、mPaaS是什么&#xff1f; mPaaS 是 Mobile Platform as a Service 的缩写&#xff0c;即移动开发平台。 蚂蚁移动开发平台mPaaS &#xff0c;融合了支付宝科技能力&#xff0c;可以为移动应用开发、测试、运营及运维提供云到端的一站式解决…...

多线BGP服务器优化实践与自动化运维方案

背景&#xff1a;企业级网络架构中的线路选择难题 在分布式业务部署场景下&#xff0c;如何通过三网融合BGP服务器实现低延迟、高可用访问&#xff1f;本文以某电商平台流量调度优化为案例&#xff0c;解析动态BGP服务器的实战价值。 技术方案设计 核心架构&#xff1a;采用…...

无法加载文件 E:\Program Files\nodejs\npm.ps1,因为在此系统上禁止运行脚本

遇到“无法加载文件 E:\Program Files\nodejs\npm.ps1,因为在此系统上禁止运行脚本”这类错误&#xff0c;通常是因为你的 PowerShell 执行策略设置为不允许运行脚本。在 Windows 系统中&#xff0c;默认情况下&#xff0c;出于安全考虑&#xff0c;PowerShell 可能会阻止运行未…...

【C++模板与泛型编程】实例化

目录 一、模板实例化的基本概念 1.1 什么是模板实例化&#xff1f; 1.2 实例化的触发条件 1.3 实例化的类型 二、隐式实例化 2.1 隐式实例化的工作原理 2.2 类模板的隐式实例化 2.3 隐式实例化的局限性 三、显式实例化 3.1 显式实例化声明&#xff08;extern templat…...

TB开拓者策略交易信号闪烁根因及解决方法

TB开拓者策略信号闪烁分析 TB开拓者策略交易信号闪烁根因 TB开拓者策略交易信号闪烁根因分析 信号闪烁是交易策略开发中常见的问题&#xff0c;特别是在TB(TradeBlazer)开拓者等平台上。以下是信号闪烁的主要根因分析&#xff1a; 主要根因 未来函数问题 使用了包含未来信息…...

什么是RDMA?

什么是RDMA&#xff1f; RDMA(RemoteDirect Memory Access)技术全称远程直接内存访问&#xff0c;就是为了解决网络传输中服务器端数据处理的延迟而产生的。它将数据直接从一台计算机的内存传输到另一台计算机&#xff0c;无需双方操作系统的介入。这允许高吞吐、低延迟的网络…...

C++面试3——const关键字的核心概念、典型场景和易错陷阱

const关键字的核心概念、典型场景和易错陷阱 一、const本质&#xff1a;类型系统的守护者 1. 与#define的本质差异 维度#defineconst编译阶段预处理替换编译器类型检查作用域无作用域&#xff08;全局污染&#xff09;遵循块作用域调试可见性符号消失保留符号信息类型安全无类…...

ASIC和FPGA,到底应该选择哪个?

ASIC和FPGA各有优缺点。 ASIC针对特定需求&#xff0c;具有高性能、低功耗和低成本&#xff08;在大规模量产时&#xff09;&#xff1b;但设计周期长、成本高、风险大。FPGA则适合快速原型验证和中小批量应用&#xff0c;开发周期短&#xff0c;灵活性高&#xff0c;适合初创企…...

【C++】嵌套类访问外部类成员

文章目录 C嵌套类访问外部类成员详解&#xff1a;权限、机制与最佳实践一、默认访问权限&#xff1a;并非友元二、访问外部类私有成员的方法1. 声明友元关系2. 通过公有接口访问 三、静态成员 vs. 非静态成员四、实际应用案例&#xff1a;Boost.Asio线程池场景需求实现关键代码…...

mac下载、使用mysql

1.如果对版本没有特别要求&#xff0c;那么直接使用brew install mysql安装即可。 2.使用 brew services start mysql 启动mysql。 3.使用 mysql -u root 登录mysql&#xff0c;这个时候还是不需要密码的 4.退出数据库&#xff1a;exit 5.给root设置一个密码&#xff0c;使用 m…...

java Lombok 对象模版和日志注解

目录 1、依赖&#xff1a; 2、在Idea中确认是否安装Lombok 插件 3、 Lombok常用注解 3.1 Getter 和 Setter 3.2 ToString 3.3 AllArgsConstructor 和 NoArgsConstructor 3.4 Data 3.5 FieldDefaults 4、 Slf4j 日志注解 4.2 日志级别 4.3 设置日志级别 1、依赖&#xff1a;…...