将成员函数作为参数传递给其他成员函数(C++ 11 <function>)

Joh*_*ski 5 c++ function member-functions std-function

假设我有一个包含三个成员函数的类,如下所示:

#include <iostream>
#include <functional>

class ClassName
{
  public:
    double add(double a, double b);
    double intermediate(double a, double b, std::function<double (double,double)> func);
    double combiner(double a, double b);
};

double ClassName::add(double a, double b)
{
  return a+b;
}

double ClassName::intermediate(double a, double b, std::function<double (double,double)> func)
{
  return func(a, b);
}

double ClassName::combiner(double a, double b)
{
  return intermediate(a, b, add);
}

int main()
{
  ClassName OBJ;
  std::cout << OBJ.combiner(12, 10);
}
Run Code Online (Sandbox Code Playgroud)

我想要做的是将成员函数"add"传递给成员函数"intermediate",然后由"combiner"调用.但是,我不认为我使用正确的语法,因为当我尝试编译它时,我收到一条错误,说"非标准语法;使用'&'创建指向成员的指针." 我不太确定会出现什么问题,因为如果这些函数不是类中的成员函数(只是命名空间中定义的常规函数​​),则此方法可以正常工作. 那么是否可以将成员函数传递给另一个成员函数?

son*_*yao 6

ClassName::add是一个非静态成员函数,ClassName需要一个实例来调用它; 它不能直接用作参数std::function<double (double,double)>.

你可以使用lambda和capture this(如@Igor Tandetnik评论的那样):

return intermediate(a, b, [this](double x, double y) { return add(x, y); } );
Run Code Online (Sandbox Code Playgroud)

或者使用std :: bind和bind this指针:

return intermediate(a, b, std::bind(&ClassName::add, this, _1, _2));
Run Code Online (Sandbox Code Playgroud)

或者创建ClassName::add静态成员函数或非成员函数(可能是因为它不使用任何成员ClassName).例如

class ClassName
{
  public:
    static double add(double a, double b);
    ...
};
Run Code Online (Sandbox Code Playgroud)


Her*_*ora 6

如果你真的想传递成员函数,你需要一个成员函数指针

class ClassName
{
   public:
    double add(double a, double b);
    using Combiner = double (ClassName::*)(double, double);
    double intermediate(double a, double b, Combiner);
    double combiner(double a, double b);
};
Run Code Online (Sandbox Code Playgroud)

这只会稍微改变intermediatecombiner

double ClassName::intermediate(double a, double b, Combiner func)
{
  return (this->*func)(a, b);
}

double ClassName::combiner(double a, double b)
{
  return intermediate(a, b, &ClassName::add);
}
Run Code Online (Sandbox Code Playgroud)