C++ 什么是仿函数 C++ 函数对象operator()重载教程【STL】

发布时间 - 2026-02-02 00:00:00    点击率:
仿函数是重载了operator()的类或结构体,本质为对象而非函数,可携带状态、支持模板推导和内联优化;用于STL时需注意public声明、const限定及严格弱序等要求。

仿函数就是重载了 operator() 的类或结构体

它不是函数,而是一个对象,但能像函数一样被调用。核心就一条:只要类里定义了 operator() 成员函数,这个类的实例就能用括号语法执行,比如 f(1, "hello")

STL 容器和算法(如 sorttransformfor_each)大量依赖仿函数,因为它们比普通函数指针更灵活——可以携带状态、支持模板参数推导、还能内联优化。

常见错误是以为“写个 operator() 就完事”,结果忘了声明为 public,或者没处理好 const 限定——比如在 std:

:set 的比较器里,operator() 必须是 const 成员函数,否则编译失败。

怎么写一个带状态的仿函数

普通函数无法保存中间数据,而仿函数可以靠成员变量记住上次调用的结果或计数。这是它最实用的特性之一。

  • operator() 必须是 public,且通常建议加 const(除非真要修改对象状态)
  • 如果用于 STL 算法(如 std::count_if),传入的是临时副本,修改成员变量不会影响原对象——这点常被忽略
  • 想持久保存状态,得确保仿函数对象本身不被复制丢失,比如用引用或 std::ref 包装

示例:

struct Adder {
    int offset = 0;
    int operator()(int x) const { return x + offset; }
};

注意:offset 是只读的,因为 operator() 声明为 const;若需要累加,去掉 const 并把 offset 改成 mutable(但 STL 一般不鼓励这种可变行为)。

std::function 和仿函数的关系

std::function 不是仿函数,而是一个通用包装器,能容纳任何可调用对象:函数指针、lambda、绑定表达式,以及真正的仿函数。

它的作用是类型擦除——把不同类型的可调用体统一成一个类型。但代价是可能有轻微性能开销(虚函数调用或小对象优化失效),所以高频场景(如循环内频繁调用)优先直接用仿函数类型,而非 std::function

容易混淆的点:

  • lambda 表达式本质是编译器生成的仿函数类,所以它天然支持 std::function 赋值,也支持直接传给 STL 算法
  • 把仿函数赋给 std::function 会触发拷贝构造,若仿函数含非拷贝成员(如 std::unique_ptr),需改用 std::ref 或改用移动语义
  • std::function 可以存 [](int){}&my_func、或 Adder{5},但三者底层实现完全不同

STL 中仿函数的实际陷阱

很多教程只讲“怎么写”,不说“在哪会崩”。真实项目里最容易栽在这些地方:

  • std::sort 要求比较仿函数是严格弱序(strict weak ordering),返回 true 表示“第一个参数排在第二个前面”。如果写成 a 或漏掉等价情况判断,会导致未定义行为甚至崩溃
  • 容器如 std::map 的模板参数是类型,不是实例,所以必须传类型名(如 std::map),而不是对象(MyCompare{}
  • 在 C++11 之前,仿函数不能有模板 operator();现在虽支持,但若用于 STL 容器模板参数,编译器可能无法推导,得显式指定
  • lambda 捕获局部变量后,若仿函数对象生命周期超过被捕获变量,就是悬垂引用——这跟普通指针问题一样隐蔽

真正难的从来不是写出来,而是让它在各种模板组合、移动语义、多线程环境下稳定工作。


# c++  # sort  # 成员变量  # 成员函数  # const  # 结构体  # int  # mutable  # 循环  # Lambda  # 指针  # 虚函数  # public  # operator  # function  # 对象  # transform  # 算法  # 能有  # 而非  # 的是  # 这是  # 第一个  # 还能  # 第二个  # 它在  # 不被  # 排在 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 什么是JavaScript解构赋值_解构赋值有哪些实用技巧  如何用腾讯建站主机快速创建免费网站?  进行网站优化必须要坚持的四大原则  企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?  Laravel如何使用Facades(门面)及其工作原理_Laravel门面模式与底层机制  Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  做企业网站制作流程,企业网站制作基本流程有哪些?  JavaScript如何实现音频处理_Web Audio API如何工作?  移动端脚本框架Hammer.js  Laravel如何实现模型的全局作用域?(Global Scope示例)  如何自定义建站之星网站的导航菜单样式?  如何在万网主机上快速搭建网站?  Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法  Python图片处理进阶教程_Pillow滤镜与图像增强  制作公司内部网站有哪些,内网如何建网站?  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  Laravel怎么创建控制器Controller_Laravel路由绑定与控制器逻辑编写【指南】  原生JS实现图片轮播切换效果  如何快速重置建站主机并恢复默认配置?  黑客如何利用漏洞与弱口令入侵网站服务器?  详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)  Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置  如何快速查询网址的建站时间与历史轨迹?  Laravel如何编写单元测试和功能测试?(PHPUnit示例)  EditPlus中的正则表达式 实战(4)  如何在局域网内绑定自建网站域名?  大连 网站制作,大连天途有线官网?  Laravel如何配置Horizon来管理队列?(安装和使用)  JS中对数组元素进行增删改移的方法总结  如何在香港服务器上快速搭建免备案网站?  jQuery中的100个技巧汇总  香港服务器部署网站为何提示未备案?  Laravel如何实现文件上传和存储?(本地与S3配置)  Thinkphp 中 distinct 的用法解析  Laravel如何为API生成Swagger或OpenAPI文档  大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?  香港服务器租用每月最低只需15元?  公司网站制作需要多少钱,找人做公司网站需要多少钱?  1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤  Laravel如何发送邮件_Laravel Mailables构建与发送邮件的简明教程  如何用免费手机建站系统零基础打造专业网站?  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  如何在阿里云香港服务器快速搭建网站?  如何在阿里云虚拟主机上快速搭建个人网站?  Laravel如何自定义分页视图?(Pagination示例)  在线制作视频的网站有哪些,电脑如何制作视频短片?  php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】  制作旅游网站html,怎样注册旅游网站?  深圳网站制作平台,深圳市做网站好的公司有哪些?