尝试使用参数包绑定 std::function 时获取未声明的参数

Getting undeclared args when trying to bind std::function using parameter pack

本文关键字:参数 std function 获取 未声明 绑定 包绑定      更新时间:2023-10-16

我正在尝试创建一个看起来像这样的可变参数模板类:

template <class... Args>
class Message
{
    private:
        std::function<void(Args...)> _function;
    public:
    //stuff
};

我想要一个绑定选项,以便用户可以在需要时绑定任何函数(带有相应的参数(。我尝试过这种方法(这个函数是 Message 类的一部分,所以Args...来自类的模板参数(:

    template<class T>
    void Message::bindFunction(void(T::*function)(Args... args), T* ownerPtr)
    {
        //This doesn't work
        _function = boost::bind(function,args, ownerPtr);
    }

但是上述方法不起作用,我尝试从绑定函数中的参数创建元组,但编译器一直说:

<args_0> undeclared identifier.

(如果我传递更多参数,它会改变(

从用户的角度来看,它应该看起来像这样:

class Foo
{
public:
    void test2(int a, float b, double c) 
    {  
        std::cout << "bla";
    }
};
int main()
{
    Foo f;
    Message<int,float,double> Mess;
    Mess.bindFunction(&Foo::test2, &f);
}

我看过很多关于转发参数的文章,但是大多数答案都让我感到困惑,而不是我已经使用不同的模板结构而不解释它们。

使用 lambda,因为它们更好(现场演示 (C++11((:

template<class T>
void bindFunction(void(T::*function)(Args...), T* ownerPtr)
{
    _function = [function, ownerPtr](Args... args)
    {
        (ownerPtr->*function)(args...);
    };
}

我们正在捕获函数指针和指向类实例的指针,并在 lambda 声明中具有[function, ownerPtr]

lambda 接受参数列表,这些参数的类型在 Message 类的可变参数模板中指定:Args... args

在内部,我们使用具有(ownerPtr->*function)语法的实例调用函数,然后通过展开参数包(args...)传入参数

要调用该函数,您可以编写如下内容:

void callFunction(Args&&... args)
{
    _function(std::forward<Args>(args)...);
}

我们使用完美转发将我们的参数传递给std::function<void(Args...)>成员。