贝利信息

C++中仿函数(Functor)是什么?(重载了括号运算符的对象)

日期:2026-01-09 00:00 / 作者:裘德小鎮的故事
仿函数是重载operator()的类对象,能像函数调用且携带状态和类型信息;普通函数无法保存上下文,而仿函数可维持成员变量实现累加、过滤等逻辑,支持STL算法并优于函数指针与lambda的复用性、内联性及类型明确性。

仿函数不是函数,是重载了 operator() 的类对象,它能像函数一样被调用,但拥有状态和类型信息。

为什么需要仿函数而不是普通函数?

普通函数无法携带上下文数据,而仿函数的实例可以保存成员变量,在多次调用间维持状态。比如实现一个累加器、带阈值的过滤器,或绑定部分参数(类似 std::bind 的早期替代)。

如何定义一个基础仿函数?

只需在类中声明并实现 operator(),参数和返回类型按需设定。注意:可重载多个 operator() 版本(不同参数列表),编译器按调用实参匹配。

struct Adder {
    int offset;
    Adder(int o) : offset(o) {}
    int operator()(int x) const { return x + offset; }
};

Adder add5(5);
int result = add5(10); // 返回 15

仿函数在 STL 算法中怎么用?

所有接受一元或二元谓词/操作的 STL 算法(如 std::for_eachstd::count_ifstd::sort)都支持仿函数作为参数。它比函数指针更灵活,比 lambda 更易复用和测试。

立即学习“C++免费学习笔记(深入)”;

struct IsEven {
    bool operator()(int n) const { return n % 2 == 0; }
};

std::vector v = {1, 2, 3, 4, 5};
int count = std::count_if(v.begin(), v.end(), IsEven{}); // 返回 2

真正容易被忽略的是生命周期和拷贝语义:STL 容器或算法可能复制你的仿函数多次,如果它持有裸指针或唯一资源,必须显式定义拷贝/移动行为,否则可能崩溃或逻辑错乱。