C++ 什么是仿函数 C++ 函数对象operator()重载教程【STL】
发布时间 - 2026-02-02 00:00:00 点击率:次仿函数是重载了operator()的类或结构体,本质为对象而非函数,可携带状态、支持模板推导和内联优化;用于STL时需注意public声明、const限定及严格弱序等要求。
仿函数就是重载了 operator() 的类或结构体
它不是函数,而是一个对象,但能像函数一样被调用。核心就一条:只要类里定义了 operator() 成员函数,这个类的实例就能用括号语法执行,比如 f(1, "hello")。
STL 容器和算法(如 sort、transform、for_each)大量依赖仿函数,因为它们比普通函数指针更灵活——可以携带状态、支持模板参数推导、还能内联优化。
常见错误是以为“写个 operator() 就完事”,结果忘了声明为 public,或者没处理好 const 限定——比如在 std: 的比较器里,
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 】
相关推荐:
Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层
如何用已有域名快速搭建网站?
如何在腾讯云服务器快速搭建个人网站?
Laravel如何配置Horizon来管理队列?(安装和使用)
如何在阿里云虚拟主机上快速搭建个人网站?
HTML透明颜色代码怎么让下拉菜单透明_下拉菜单透明背景指南【技巧】
极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?
Laravel怎么生成URL_Laravel路由命名与URL生成函数详解
Python文件异常处理策略_健壮性说明【指导】
Android自定义listview布局实现上拉加载下拉刷新功能
5种Android数据存储方式汇总
焦点电影公司作品,电影焦点结局是什么?
jQuery validate插件功能与用法详解
教学论文网站制作软件有哪些,写论文用什么软件
?
Laravel怎么实现模型属性转换Casting_Laravel自动将JSON字段转为数组【技巧】
西安专业网站制作公司有哪些,陕西省建行官方网站?
手机网站制作平台,手机靓号代理商怎么制作属于自己的手机靓号网站?
Laravel如何处理表单验证?(Requests代码示例)
网站制作软件有哪些,制图软件有哪些?
手机网站制作与建设方案,手机网站如何建设?
Swift开发中switch语句值绑定模式
Laravel如何生成API文档?(Swagger/OpenAPI教程)
php在windows下怎么调试_phpwindows环境调试操作说明【操作】
如何在局域网内绑定自建网站域名?
如何在万网自助建站平台快速创建网站?
Python3.6正式版新特性预览
Laravel如何实现邮件验证激活账户_Laravel内置MustVerifyEmail接口配置【步骤】
如何在云虚拟主机上快速搭建个人网站?
如何在万网自助建站中设置域名及备案?
Laravel集合Collection怎么用_Laravel集合常用函数详解
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
详解vue.js组件化开发实践
googleplay官方入口在哪里_Google Play官方商店快速入口指南
Laravel如何为API编写文档_Laravel API文档生成与维护方法
Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】
EditPlus中的正则表达式 实战(2)
Laravel如何记录自定义日志?(Log频道配置)
Laravel如何发送系统通知?(Notification渠道示例)
Laravel distinct去重查询_Laravel Eloquent去重方法
Laravel如何实现用户注册和登录?(Auth脚手架指南)
高防服务器租用首荐平台,企业级优惠套餐快速部署
Laravel如何实现API版本控制_Laravel API版本化路由设计策略
Laravel如何处理和验证JSON类型的数据库字段
如何快速配置高效服务器建站软件?
Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理
Claude怎样写结构化提示词_Claude结构化提示词写法【教程】
香港服务器部署网站为何提示未备案?
Laravel如何实现用户密码重置功能?(完整流程代码)
历史网站制作软件,华为如何找回被删除的网站?
如何用AI帮你把自己的生活经历写成一个有趣的故事?

