我有一个接受回调的函数。它应该与函数ptrs,lambdas(无状态和具有状态)一起使用,我可以执行以下操作:
template<typename t_func>
void add_command(const std::string& name, t_func func)
问题是我需要使用func的参数类型。所以我这样做了:
template<typename... t_args>
void add_command(const std::string& name, const std::function<void(t_args...)>& args)
这会产生以下错误:no matching function for call to
...
note: template argument deduction/substitution failed
在仍然可以访问通用参数类型的参数的同时,有什么方法可以传递它吗?我知道std::result_of
,有没有类似的std::arguments_of
?
答案 0 :(得分:6)
std::function
是类型擦除模板。类型推导与类型擦除是相反的(几乎是相反的)。
推导类型擦除模板的类型是代码气味。而且很少起作用。
在c++17中有一个演绎指南,因此您可以这样做:
template<typename... t_args>
void add_command(const std::string& name, const std::function<void(t_args...)>& args)
void add_command(const std::string& name, t_func const& func) {
std::function f = func;
add_command(name, f);
}
这是不完美的,但不可能找到完美的解决方案。
推论指南look like:
template<class R, class... ArgTypes>
function(R(*)(ArgTypes...)) -> function<R(ArgTypes...)>;
template<class F>
function(F) -> function</*see below*/>;
通过检查function
提取&F::operator()
的签名。这可能因重载或模板而失败。当然,这不适用于重载的函数名称。
您可以使用函数特征类在c++11中复制它:
template<class X>
struct function_traits:function_traits<decltype(&X::operator())> {};
#define MEM_FUN_HELPER2(...) \
template<class R, class T, class...Args> \
struct function_traits<R(T::*)(Args...) __VA_ARGS__>:function_traits<R(Args...)>{}; \
template<class R, class T, class...Args> \
struct function_traits<R(T::*)(Args..., ...) __VA_ARGS__>:function_traits<R(Args..., ...)>{}; \
template<class R, class T, class...Args> \
struct function_traits<R(T::*)(Args...) __VA_ARGS__ noexcept>:function_traits<R(Args...) noexcept>{}; \
template<class R, class T, class...Args> \
struct function_traits<R(T::*)(Args..., ...) __VA_ARGS__ noexcept>:function_traits<R(Args..., ...) noexcept>{}
#define MEM_FUN_HELPER1(...) \
MEM_FUN_HELPER2(__VA_ARGS__); \
MEM_FUN_HELPER2(__VA_ARGS__ &); \
MEM_FUN_HELPER2(__VA_ARGS__ &&)
#define MEM_FUN_HELPER0(...) \
MEM_FUN_HELPER1(__VA_ARGS__); \
MEM_FUN_HELPER1(const __VA_ARGS__)
#define MEM_FUN_HELPER() \
MEM_FUN_HELPER0(); \
MEM_FUN_HELPER0(volatile)
MEM_FUN_HELPER();
template<class R, class...Args>
struct function_traits<R(*)(Args...)>:function_traits<R(Args...)>{};
template<class R, class...Args>
struct function_traits<R(*)(Args..., ...)>:function_traits<R(Args..., ...)>{};
template<class R, class...Args>
struct function_traits<R(*)(Args...) noexcept>:function_traits<R(Args...) noexcept>{};
template<class R, class...Args>
struct function_traits<R(*)(Args..., ...) noexcept>:function_traits<R(Args..., ...) noexcept>{};
template<class R, class...Args>
struct function_traits<R(Args...) noexcept> : function_traits<R(Args...)> {
enum {is_noexcept=true};
};
template<class R, class...Args>
struct function_traits<R(Args..., ...) noexcept> : function_traits<R(Args..., ...)> {
enum {is_noexcept=true};
};
template<class R, class...Args>
struct function_traits<R(Args...)> {
template<template<class...>class Z>
using transcribe=Z<R(Args...)>;
using std_function = transcribe<std::function>;
using result_type = R;
using arg_tuple = std::tuple<Args...>;
enum{is_noexcept=false};
};
template<class R, class...Args>
struct function_traits<R(Args..., ...)> {
template<template<class...>class Z>
using transcribe=Z<R(Args..., ...)>;
using std_function = transcribe<std::function>;
using result_type = R;
// doesn't really work, but what ya gonna do:
using arg_tuple = std::tuple<Args...>;
enum{is_noexcept=false};
};
这太疯狂了; MEM_FUN_HELPER();
扩展为48个模板专长。 3个ref限定词(&
,&&
,什么也没有),然后是其他4种事物(const
,volatile
,noexcept
和...
C风格varargs),必须“手动”处理。
无论如何,一旦有了,就可以做到:
template<typename... t_args>
void add_command(const std::string& name, const std::function<void(t_args...)>& args)
template<class t_func>
void add_command(const std::string& name, t_func const& func) {
typename function_traits<t_func>::std_function f = func;
add_command(name, f);
}
在c++11中。 (粗略且不完全)这与c++17推论指南相同。