当我boost::fusion::fused
在使用 decltype 时查看函数包装器中的错误时,就会出现这种情况。问题似乎是无效的 decltype 是编译错误,即使不会使用需要它的模板实例化,我也不知道如何解决这个问题以创建通用函数包装器。
这是我对单参数包装器的尝试:
#include <utility>
#include <type_traits>
template <class T>
typename std::add_rvalue_reference<T>::type declval();
template <class Fn, class Arg>
struct get_return_type
{
typedef decltype(declval<Fn>()(declval<Arg>())) type;
};
template <class Fn>
struct wrapper
{
explicit wrapper(Fn fn) : fn(fn) {}
Fn fn;
template <class Arg>
typename get_return_type<Fn,Arg&&>::type
operator()(Arg&& arg)
{
return fn(std::forward<Arg>(arg));
}
template <class Arg>
typename get_return_type<const Fn,Arg&&>::type
operator()(Arg&& arg)
{
return fn(std::forward<Arg>(arg));
}
};
问题是,这不适用于非 const 版本的参数不能转换为 const 版本的参数的情况。例如:
#include <iostream>
struct x {};
struct y {};
struct foo
{
void operator()(x) { std::cout << "void operator()(x)" << std::endl; }
void operator()(y) const { std::cout << "void operator()(y) const" << std::endl; }
};
int main()
{
wrapper<foo> b = wrapper<foo>(foo());
b(x()); // fail
}
在我看来,由 引起的 decltype 表达式的失败void operator()(y) const
应该只是导致该函数由于 SFINAE 而被删除。