首页 > 解决方案 > 如何制作可以包装任何函数的函数包装器?

问题描述

#include <iostream>
#include <string>

template<typename Func>
class FuncWrapper {
    Func func;
    std::string name;

public:

    FuncWrapper(Func func, std::string name) : func(func), name(std::move(name)) {}

    template<typename ...Args>
    auto operator()(Args ...args) {
        std::cout << "Entering " + name + '\n';
        auto ret = func(args...);
        std::cout << "Exiting " + name + "\n\n";
        return ret;
    }
};

template<typename Func>
auto makeFuncWrapper(Func func, std::string name) {
    return FuncWrapper<Func>(func, name);
}

int main() {

    auto helloWorld = []() { std::cout << "Hello World\n"; };
    auto addTwoNums = [](int a, int b) { std::cout << "Adding numbers...\n"; return a + b; };

    // makeFuncWrapper(helloWorld, "helloWorld")(); // error: 'void ret' has incomplete type.
    std::cout << makeFuncWrapper(addTwoNums, "addTwoNums")(4, 5) << '\n';

}

这个类FuncWrapper工作正常,并为传递的函数添加了一个额外的功能,直到传递的函数是一个返回的函数void。我收到类型ret不完整的错误。即使使用返回类型为 的函数,还有其他方法可以使其工作void吗?我知道我不能有void变量,但在这里,唯一的目的ret是在函数完成后返回,并且不会以错误的方式使用它。是否有解决方案或一些解决方法可以使其工作?实现包装任何函数的函数包装器的任何更好方法?

标签: c++templateswrappervoid

解决方案


您可能会使用 Raii:

template<typename ...Args>
auto operator()(Args ...args) {
    struct EndScope {
        std::string& name;
        ~EndScope() { std::cout << "Exiting " + name + "\n\n"; }
    } endScope(name);

    std::cout << "Entering " + name + '\n';
    return func(args...);
}

您可能会进一步处理异常std::uncaught_exceptions

请参阅raii-way-to-get-errors-that-are-caught-during-destruction


推荐阅读