gpt4 book ai didi

c++ - 如何在 C++17 中制作模板包装器/装饰器

转载 作者:行者123 更新时间:2023-12-02 09:48:23 27 4
gpt4 key购买 nike

亲爱的 Stackoverflow 社区,

我对 C++ 还是有点陌生​​,我一直在摸不着头脑,还没有找到解决我问题的方法。我一直在搜索和尝试一些东西,现在我已经到了提出问题会更有益和更有教育意义的地步。

问题:

我想创建一个类或函数来包装/修饰带或不带参数的给定函数。就像 python 或 c# 之类的老式 @wrapthis。

到目前为止我发现的最接近的东西(优雅、简短且易于使用)来自这个 stackoverflow 答案:C++ Equivalent Decorator

挠头的部分是试图传递一个指针函数。我收到的错误:

Error (active) E0300 a pointer to a bound function may only be used to call the function

这显然意味着不允许以某种方式以这种方式传递指针,那么我在这里有哪些选择?

一个有效的例子作为答案会很棒!

下面是我想要实现的示例:

Something.h

class Something
{
private:
public:
Something() {}
void v_func_with_nothing() { std::cout << "v_func_with_nothing" << "\n"; }
void v_func_with_void(void) { std::cout << "v_func_with_void" << "\n"; }
void v_func_with_one_arg(int x) { std::cout << "v_func_with_one_arg" << x << " " << "\n"; }
void v_func_with_args(int x, int y) { std::cout << "v_func_with_args" << x << " " << y << "\n"; }

int return_func_with_nothing() { return 1; }
int return_func_with_void(void) { return 3; }
int return_func_with_one_arg(int x) { return x; }
int return_func_with_args(int x, int y) { return x+y; }
};

Decorator.h [再次来源:C++ Equivalent Decorator ]

template<typename T>
auto decorator(T&& func)
{
auto new_function = [func = std::forward<T>(func)](auto&&... args)
{
std::cout << "BEGIN decorating...\n";
auto result = func(std::forward<decltype(args)>(args)...);
std::cout << "END decorating\n";
return result;
};
return new_function;
}

main.cpp

#include <iostream>
#include "Something.h"
#include "Decorator.h"

int main()
{
Something* something = new Something();
auto somedeco = decorator(&something->return_func_with_one_arg);//<-- error here in argument
//int value = somedeco(**enter an argument**);
//std::cout << value << "\n";
return 0;
}

谢谢!

编辑:解决方案

根据下面给出的友好回答,我想用一个例子来编辑这篇文章。问题的解决方案是使用 lambda。

Decorator.h:我创建了 2 个装饰器(一个用于返回函数,一个用于 void 函数):

template<typename T>
auto DECO_R(T&& func)
{
try
{
auto new_function = [func = std::forward<T>(func)](auto&&... args)
{
std::cout << "BEGIN RETURN decorating...\n";
auto result = func(std::forward<decltype(args)>(args)...);
std::cout << "END RETURN decorating\n";
return result;
};
return new_function;
}
catch (const std::exception& ex)
{
std::cout << ex.what() << "\n";
}
}

template<typename T>
auto DECO_V(T&& func)
{
try
{
auto new_function = [func = std::forward<T>(func)](auto&&... args)
{
std::cout << "BEGIN VOID decorating...\n";
func(std::forward<decltype(args)>(args)...);
std::cout << "END VOID decorating\n";
};
return new_function;
}
catch (const std::exception& ex)
{
std::cout << ex.what() << "\n";
}
}

Main.cpp:2 个例子

int main()
{
Something* something = new Something();

auto somedeco = DECO_R(
[&](int x) {
return something->return_func_with_one_arg(x);
});

int value = somedeco(255);
std::cout << value << "\n";

auto some_v_deco = DECO_V(
[&](int x) {
return something->v_func_with_one_arg(x);
});

some_v_deco(2);

return 0;
}

输出

开始返回装饰...
结束返回装饰
255

开始作废装饰...
v_func_with_one_arg2
END VOID装饰

我希望这对其他人有帮助。

最佳答案

调用 decorator(&something->return_func_with_one_arg) 无效。在 C++ 中没有指向绑定(bind)函数的指针。

例如,如果您希望 somedeco 成为一个类似函数的对象来包装对 something->return_func_with_one_arg(42) 的调用,您将需要包装在 lambda 中调用:

auto somedeco = decorator(
[&]() {
return something->return_func_with_one_arg(42);
}
);
somedeco();

或者你可以通过装饰器传递参数:

auto somedeco = decorator(
[&](int x) {
return something->return_func_with_one_arg(x);
}
);
somedeco(42);

请记住,这将要求 something 指向的对象比 decorator 返回的对象活得更久。

关于c++ - 如何在 C++17 中制作模板包装器/装饰器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62893060/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com