【问题标题】:How to write a template wrapper method for other class member functions?如何为其他类成员函数编写模板包装器方法?
【发布时间】:2019-10-29 13:12:43
【问题描述】:

我尝试为具有不同参数的不同函数创建一个模板化包装器。 设置是一个类A,基本实现了两个方法foobar。另一个类B 将包装这些方法并添加新功能。

以下链接中的解决方案非常适用于非类函数: c++11: Templated wrapper function

但是如果我尝试调用另一个类的方法,我会得到一个错误。

#include <algorithm>
#include <functional>
#include <iostream>

class A
{
public:
    void foo(int x) {
        std::cout << "Foo: " << x << std::endl;
    }

    void bar(int x, float y) {
        std::cout << "Bar: " << x << ", " << y << std::endl;
    }
};

class B
{
public:
    void fooAndMore(int x) {
        foobarWrapper(&A::foo, 1);
    }

    void barAndMore(int x, float y) {
        foobarWrapper(&A::bar, 1, 3.5f);
    }

    template<typename  T, typename... Args>
    void foobarWrapper(T&& func, Args&&... args)
    {
        std::cout << "Start!" << std::endl;
        std::forward<T>(func)(std::forward<Args>(args)...);
        std::cout << "End!" << std::endl;
    }
};

int main()
{
    B b;
    b.fooAndMore(1);
    b.barAndMore(2, 3.5f);
}

我期待这样的事情:

Start!
Foo: 1
End!
Start!
Bar: 1, 3.5
End!

但是我得到了:

error C2064: term does not evaluate to a function taking 1 arguments
note: see reference to function template instantiation 'void B::foobarWrapper<void(__thiscall A::* )(int),int>(T &&,int &&)' being compiled
    with
    [
        T=void (__thiscall A::* )(int)
    ]

error C2064: term does not evaluate to a function taking 2 arguments
note: see reference to function template instantiation 'void B::foobarWrapper<void(__thiscall A::* )(int,float),int,float>(T &&,int &&,float &&)' being compiled
    with
    [
        T=void (__thiscall A::* )(int,float)
    ]

知道如何解决这个问题吗?

【问题讨论】:

  • 我还有一个与此主题相关的问题:是否可以获取传递函数的名称?我试过:``` auto funcName = typeid(func).name(); std::cout

标签: c++ class c++11 templates c++17


【解决方案1】:

最简单的解决方法是将类A 的成员函数设为static(See online)

class A
{
public:
    static void foo(int x) {
    ^^^^^^
        std::cout << "Foo: " << x << std::endl;
    }

    static void bar(int x, float y) {
    ^^^^^^
        std::cout << "Bar: " << x << ", " << y << std::endl;
    }
};

否则,您需要传递类A 的实例,以在foobarWrapper 函数中调用其成员函数。使用 lambda,您可以将它们打包到可调用的 func 并传递给 foobarWrapper

(See online)

class B
{
public:
    void fooAndMore(const A& a_obj, int x) {
        foobarWrapper([&]() { return a_obj.foo(x); });
        //            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^Args captured to the lambda
    }

    void barAndMore(const A& a_obj, int x, float y) {
        foobarWrapper([&]() { return a_obj.bar(x, y); });
        //            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^  Args captured to the lambda
    }

    template<typename  T>
    void foobarWrapper(T&& func)   // no Args needed any more (@credits Jarod42)
    {
        std::cout << "Start!" << std::endl;
        std::forward<T>(func)();   // simply call the func
        std::cout << "End!" << std::endl;
    }
};

int main()
{
    B b;
    b.fooAndMore(A{}, 1);       // pass a temporary A object
    b.barAndMore(A{}, 2, 3.5f);
}

【讨论】:

  • 感谢您的快速响应。在我的情况下,静态不是一个选项,但我将尝试获取具有基本实现的类的实例。谢谢 :)
  • @boss0r 我在更新的答案中添加了第二个选项。
  • 再次感谢你,你帮了我很多 :) 这很好,完美!
  • @Jarod42 是的......你是对的。我已相应更新。感谢您的建议。
【解决方案2】:

试试这个,

#include <algorithm>
#include <functional>
#include <iostream>

class A
{
public:
    void foo(int x) {
        std::cout << "Foo: " << x << std::endl;
    }

    void bar(int x, float y) {
        std::cout << "Bar: " << x << ", " << y << std::endl;
    }
};

class B
{
public:
    void fooAndMore(int x) {
        foobarWrapper(&A::foo, x);
    }

    void barAndMore(int x, float y) {
        foobarWrapper(&A::bar, x, y);
    }

    template<typename  T, typename... Args>
    void foobarWrapper(T func, Args&&... args)
    {
        std::cout << "Start!" << std::endl;

        auto caller = std::mem_fn( func); // Newly added lines
        caller( A(), args...);  // Newly added line

        std::cout << "End!" << std::endl;
    }
};

int main()
{
    B b;
    b.fooAndMore(1);
    b.barAndMore(2, 3.5f);
}

输出:

Start!
Foo: 1
End!
Start!
Bar: 2, 3.5
End!

查看此链接了解更多详情std::mem_fn

【讨论】:

  • 谢谢!我以前没有 std::mem_fn 这个。
  • @boss0r 欢迎。
猜你喜欢
  • 2018-06-29
  • 1970-01-01
  • 1970-01-01
  • 2020-05-24
  • 1970-01-01
  • 2017-03-10
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多