Обходной путь, позволяющий tr1::function проглотить возвращаемые значения
В качестве продолжения может ли tr1::function проглотить возвращаемые значения? Как можно обойти ограничение, которое tr1::function
не может проглотить возвращаемые значения?
Это должно работать в конкретном случае проглатывания возвращаемого значения вызываемого объекта без аргументов:
template<typename FuncT>
struct swallow_return_t
{
explicit swallow_return_t(FuncT i_Func):m_Func(i_Func){}
void operator()(){ m_Func(); }
FuncT m_Func;
};
template<typename FuncT>
swallow_return_t<FuncT>
swallow_return(FuncT f)
{
return swallow_return_t<FuncT>(f);
}
Тогда используйте как:
int Foo();
std::tr1::function<void()> Bar = swallow_return(Foo);
Я предполагаю, что вариационные шаблоны и совершенная пересылка позволят обобщить эту технику на произвольные списки параметров. Есть ли способ лучше?
1 ответ
В GCC 4.6.1 у меня работает следующее:
#include <functional>
int foo() { return 5; }
int goo(double, char) { return 5; }
int main()
{
std::function<void()> f = foo;
std::function<void(float, int)> g = goo;
(void)f();
(void)g(1.0f, 'a');
}
Вот обертка, использующая лямбды, но она пока не автоматическая
template <typename T, typename ...Args>
struct strip_return
{
static inline std::function<void(Args...)> make_function(std::function<T(Args...)> f)
{
return [&f](Args... args) -> void { f(args...); };
}
};
int main()
{
auto q = strip_return<int>::make_function(std::bind(foo));
q();
}
Забудьте среднюю часть. Хорошо, так как std::function
это стирание типов, трудно понять базовые типы. Однако, если вы перейдете непосредственно к ссылке на функцию, вы можете полностью избежать этих проблем:
template <typename T, typename ...Args>
static inline std::function<void(Args...)> make_direct(T (&f)(Args...))
{
return [&f](Args... args) -> void { f(args...); };
}
int main()
{
auto p = make_direct(foo);
q();
}