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

C++ 函数对象、仿函数与 Lambda 表达式详解

C++ 函数对象、仿函数与 Lambda 表达式详解

在 C++ 中,函数对象(Function Object)仿函数(Functor)Lambda 表达式 是三种实现可调用行为的技术,它们在功能上类似,但语法和适用场景有所不同。


一、函数对象(仿函数)

函数对象 是通过重载 operator() 的类实例,使对象可以像函数一样被调用。它本质是类对象,但行为类似函数,因此也被称为 仿函数

1. 基本用法
class Adder {
public:Adder(int n) : num(n) {}int operator()(int x) const {return x + num;}
private:int num;
};int main() {Adder add5(5);cout << add5(3); // 输出 8
}
  • Adder 类重载了 operator(),实例 add5 可以像函数一样调用。
  • 优点:可以保存状态(如成员变量 num),比函数指针更灵活。
2. 应用场景
  • STL算法:如 std::sortstd::transform 需要传入自定义逻辑时。
  • 需要状态的函数:例如计数器、累加器等。
  • 模板元编程:作为策略类传递。

二、Lambda 表达式

Lambda 表达式 是 C++11 引入的匿名函数,语法简洁,可捕获上下文变量。编译器会将其转换为匿名函数对象。

1. 基本语法
[捕获列表](参数列表) -> 返回类型 { 函数体 }
  • 捕获列表:指定如何捕获外部变量(值捕获 [x]、引用捕获 [&x]、隐式捕获 [=][&])。

  • mutable:允许修改按值捕获的变量(默认 const)。

  • 示例

    auto add5 = [num=5](int x) { return x + num; };
    cout << add5(3); // 输出 8
    
2. 底层实现

Lambda 会被编译为一个匿名类,类似仿函数:

// 编译器生成的类
class __AnonymousLambda {
public:__AnonymousLambda(int n) : num(n) {}int operator()(int x) const { return x + num; }
private:int num;
};
3. 高级特性
  • C++14:支持泛型 Lambda(参数用 auto)。

    auto print = [](auto x) { cout << x; };
    
  • C++20:支持模板 Lambda 和 [=, this] 显式捕获 this


三、函数对象 vs Lambda 表达式
特性函数对象Lambda 表达式
语法复杂度需要显式定义类内联匿名,语法简洁
状态管理通过成员变量保存状态通过捕获列表管理外部变量
重载 operator()支持多个重载版本单一函数体,不支持重载
性能通常可内联,高效同函数对象,编译器优化
适用场景复杂逻辑、需要复用的操作简单、一次性使用的逻辑

四、选择建议
  • 使用 Lambda:当逻辑简单、临时使用,或需要捕获局部变量时。
  • 使用函数对象:当需要复用逻辑、维护复杂状态,或需要重载 operator() 时。

五、示例对比
1. 排序比较器

仿函数实现

struct Compare {bool operator()(int a, int b) const { return a > b; }
};
vector<int> vec {3, 1, 4};
sort(vec.begin(), vec.end(), Compare());

Lambda 实现

sort(vec.begin(), vec.end(), [](int a, int b) { return a > b; });
2. 带状态的累加器

仿函数实现

class Counter {int count = 0;
public:int operator()() { return ++count; }
};
Counter cnt;
cout << cnt() << cnt(); // 输出 1 2

Lambda 实现

auto counter = [count=0]() mutable { return ++count; };
cout << counter() << counter(); // 输出 1 2

六、总结
  • 函数对象/仿函数:适合复杂、可复用的逻辑,或需要重载的场景。
  • Lambda 表达式:适合简单、临时的操作,尤其是需要捕获上下文时。
  • 底层一致性:Lambda 本质是编译器生成的函数对象,二者性能相同。
http://www.xdnf.cn/news/7349.html

相关文章:

  • 12.vue整合springboot首页显示数据库表-实现按钮:【添加修改删除查询】
  • 深入Java G1 GC调优:如何解决高延迟与吞吐量瓶颈
  • 嵌入式学习笔记 - STM32独立看门狗IWDG与窗口看门狗WWDG的区别
  • HTTPS实验室——TLS/TLCP一站式解决方案
  • C语言——深入理解指针(一)
  • rosbag使用记录
  • 搭建一个永久免费的博客
  • Java设计模式之组合模式:从入门到精通(保姆级教程)
  • Java 泛型详解
  • 黄仁勋Computex演讲:将于三季度推出下一代GB300系统,个人AI计算机DGX Spark已全面投产
  • 进程和线程有什么区别?多线程有什么优缺点?线程的创建方式有哪些?如何简单的使用线程?用户线程和守护线程有什么区别?start 和 run 方法有什么区别?
  • go 与面向对象编程(OOP)
  • 设置IDEA打开新项目使用JDK17
  • 【OpenCV基础2】图像运算、水印、加密、摄像头
  • 信号量基础入门:并发控制的核心概念
  • BGP选路
  • 常用ECSQL整理
  • ‌AT6558R-5N22北斗B1I单频导航芯片
  • 《深入理解数组名:sizeof(arr)、arr 和 arr 的区别》
  • 三种嵌入式开发常用的组网方式
  • 【C++】C++的IO流
  • 青岛地铁二号线列车运行图优化系统
  • AIGC与文本生成:人工智能写作的新纪元
  • Adminer:一个基于Web的轻量级数据库管理工具
  • Python | 需求预测模型
  • 使用 docker-volume-backup 备份 Docker 卷
  • plc基础知识整理(三菱)
  • PHP 实现连续子数组的最大和、整数中1出现的次数
  • 详解Oracle HASH CHAIN和HASH BUCKET
  • TS04:高性能四通道自动灵敏度校准电容触摸传感器