c++ - 将带有可变参数的模板函数分配给函数指针
问题描述
在 C++ 中,我试图制作接受第一个参数并在其上调用方法的转发包装器。这最终出现在wrapper
给定代码中的方法中。这在调用它时工作正常。
但是我想将此模板化包装器分配给函数指针。在我的用例中,函数指针是给定的,我无法将其更改为 astd::function
或类似的东西。这是因为它在 C api 中使用的地方。
我确实创建了以下示例:
#include <iostream>
template<auto FUNC, typename T, typename ... Params>
static auto wrapper(void* handle, Params&& ... args) {
auto* ser = static_cast<T*>(handle);
return (ser->*FUNC)(std::forward<Params>(args)...);
}
class MyTestClass {
public:
int method1(int i, int j) { return i + j; }
float method2(float f) {
return f * 2;
}
};
int main() {
MyTestClass thing{};
int (*serialize)(void* handle, int i, int j);
serialize = wrapper<&MyTestClass::method1, MyTestClass>;
auto callMethod1Result = wrapper<&MyTestClass::method1, MyTestClass>(&thing, 1, 2);
std::cout << "callMethod1Result: " << callMethod1Result << std::endl;
return 0;
}
该方法的调用工作正常,但是:
int (*serialize)(void* handle, int i, int j);
serialize = wrapper<&MyTestClass::method1, MyTestClass>;
不起作用,给我错误:
/.../temp.cpp:23:17: error: no matches converting function ‘wrapper’ to type ‘int (*)(void*, int, int)’
serialize = wrapper<&MyTestClass::method1, MyTestClass>;
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
/.../temp.cpp:4:13: note: candidate is: ‘template<auto FUNC, class T, class ... Params> auto wrapper(void*, Params&& ...)’
static auto wrapper(void* handle, Params&& ... args) {
^~~~~~~
经过一番尝试,我确实发现该Params&& ... args
部分导致了问题,因为如果我制作一个没有可变参数的更明确的包装器,那么它确实有效。
我的主要问题是:我可以将带有可变参数的模板化方法分配给函数指针,以及如何分配?
解决方案
问题是wrapper
作为args
转发引用,它的类型总是引用:左值引用或右值引用。但是serialize
被声明为函数指针取值i
和j
类型的值int
,它们不能匹配引用类型,这使得模板参数推导Params
失败serialize = wrapper<&MyTestClass::method1, MyTestClass>;
。
i
您可以通过更改类型和j
声明来修复它serialize
。
例如
int (*serialize)(void* handle, int&& i, int&& j);
serialize = wrapper<&MyTestClass::method1, MyTestClass>;
或按价值改变wrapper
。args
例如
template<auto FUNC, typename T, typename ... Params>
static auto wrapper(void* handle, Params ... args)
推荐阅读
- flutter - 包裹在一行内被切割
- c# - 与三角形背面碰撞时如何忽略相机碰撞?
- python-3.x - 在 qml 中禁用最小化功能
- mysql - 如何在mysql中将日文日期格式转换并更新为真实的日期时间格式
- c# - 带有身份的 Blazor WebAssembly 的 SignalR Hub 授权
- d3.js - dc.ChoroplethChart 的问题
- rust - 在 rust 中重新编码 C++ 任务队列。期货是正确的抽象吗?
- linux - 使用热键后如何在 vim 中转到行尾?
- android - 启用 FLAG_LAYOUT_NO_LIMITS 时,选项子菜单会离开屏幕
- javascript - 如何在 JavaScript 中应用 display: flex 和 flex-direction: row?