带有任何参数的模板函数
Template functor with any parameters
我正在尝试创建模板函数,它将用任何数量的参数作为参数对象和成员函数。我不知道如何用模板正确编写代码。
template<typename ItemT,
class T,
typename ...Args>
struct Builder
{
ItemT operator()(T& object, ItemT (T::*method)(Args...), Args && ... args)
{
return (object.*method)(std::forward<Args>(args)...);
}
};
struct Object
{
int method(int, int, int) { return 4; }
};
int main()
{
Object obj;
Builder<int, Object>()(obj, &Object::method); // Error here
}
如果我制作没有参数的object ::方法 - 代码编译。但是使用参数 - 否。
严重性代码描述项目文件行抑制状态错误c2664'int Builder :: operator()(t&amp;,itemt(_____thiscall object ::*)(void))':无法转换参数2从'int(____thiscall object ::*)(int,int,int,int)'to'int(__ thiscall object ::*)(void)'草稿c: drafds main.cpp 139
假设您不想更改Builder
的当前定义,这就是您需要实例化的方式:
Builder<int, Object, int, int, int>()(obj, &Object::method, 0, 0, 0);
// ^ ^ ^^^^^^^^^^^^^ ^^^^^^^
// ItemT | | |
// T Args... args...
operator()
调用中的args...
参数扩展必须与传递的TArgs...
包与Builder
本身匹配。
wandbox示例
这是替代不太严格设计:
template<typename T>
struct Builder
{
template <typename TFnPtr, typename... Args>
auto operator()(T& object, TFnPtr method, Args && ... args)
{
return (object.*method)(std::forward<Args>(args)...);
}
};
上述Builder
可以像这样使用:
int main()
{
Object obj;
Builder<Object>()(obj, &Object::method, 0, 0, 0);
}
在这种情况下,成员函数指针的类型是通过TFnPtr
推导的,并且不受任何特定参数集的约束。variadic参数不再是Builder
的一部分 - 它们是Builder::operator()
的一部分,因此可以推导并转发为(object.*method)
。
wandbox示例
您可以避免在Builder
上完全避免模板,仅依赖模板参数扣除:
struct Builder {
template <typename Obj, typename R, typename ... FArgs, typename ... Args>
R operator()(Obj& obj, R (Obj::*fn)(FArgs...), Args&&... args) {
return (obj.*fn)(std::forward<Args>(args)...);
}
};
我选择使用两个参数包来允许完美的转发:operator()
调用的值类别不一定与目标方法匹配。这还允许在应用成员函数指针时进行参数的隐式转换。请注意,此实现将与Obj
的const
方法不匹配。
您当然可以使用auto
返回类型(C 14)或落后返回类型(C 11)放松一下。由于Vittorio的答案已经向您介绍了C 14方法,因此我将解决后者。operator()
然后变为:
template <typename Obj, typename FnPtr, typename ... Args>
auto operator()(Obj& obj, FnPtr fn, Args&&... args)
-> decltype((obj.*fn)(std::forward<Args>(args)...)) {
return (obj.*fn)(std::forward<Args>(args)...);
}
然后,用法将简单地为:
Object obj;
Builder()(obj, &Object::method, 0, 0, 0);
live demo on Coliru。
- static_assert在宏中,但也可以扩展到可以用作函数参数的东西
- C++中的高效循环缓冲区,它将被传递给C样式数组函数参数
- 当从函数参数中的临时值调用复制构造函数时
- 如何从"decltype()"获取函数参数的数量<funtion>?
- 如何将lambda作为模板类的成员函数参数
- 模板参数推导失败,函数参数/参数不匹配
- 如何在C++中将迭代器作为函数参数传递
- 将函数参数"const char*"转换为"std::string_view"是
- C++ 如何将数组值解压缩为函数参数
- 主函数参数的属性
- 具有两个间接寻址运算符 (C++) 的函数参数的用途
- "Warning: Comma within array index expression"但逗号分隔函数参数
- 如何定义在用作函数参数时工作的类模板的转换
- 将函数参数完美转发到函数指针:按值传递呢?
- 为什么我不能将引用作为 std::async 的函数参数传递
- 什么..(省略号)作为函数原型中唯一的函数参数,C++?
- 是否可以就地构造一个固定大小的数组作为函数参数?
- 接受模板作为函数参数
- 将成员函数作为构造函数参数调用时出错 "Variable is not a type name"
- Arduino 函数参数