1

我有一个可调用类型的元组。

std::tuple<std::function<int(int)>, std::function<std::string(double)>> funcs;

我想创建另一个具有每个可调用结果类型的元组。例如,funcs包含int->intdouble->std::string

我怎样才能创建一个results依赖于每个元素的元组,funcs看起来像这样。

std::tuple<int, std::string> results;
4

3 回答 3

1
#include <tuple>
#include <functional>
#include <string>


// takes an arbitrary tuple of std::functions and creates a
//   tuple of the return types of those std::functions
template<class T, class... Types>
struct TupleHandler;

// handles the recursive inheritance base case when all the
//   elements of the tuple have been processed
template<class... Types>
  struct TupleHandler<std::tuple<>, Types...> {
    using ReturnTypeTuple = std::tuple<Types...>;
  };

// Strips off the first std::function in the tuple, determines
//   its return type and passes the remaining parts on to have the next element
//   processed
template<class Return, class... Rest, class... Tail, class... Types>
struct TupleHandler<std::tuple<std::function<Return(Rest...)>, Tail...>, Types...> : TupleHandler<std::tuple<Tail...>, Types..., Return> {
     using ReturnTypeTuple = typename TupleHandler<std::tuple<Tail...>, Types...,  Return>::ReturnTypeTuple;
};

int main()
{
    std::tuple<std::function<int(int)>, std::function<std::string(double)>> funcs;

    // Of course for this simple use case you could have just used std::make_tuple, but it still demonstrates the solution
    TupleHandler<decltype(funcs)>::ReturnTypeTuple return_value_tuple(std::get<0>(funcs)(1), std::get<1>(funcs)(4.4));

    // added per comment
    auto x = [](auto funcs){ typename TupleHandler<decltype(funcs)>::ReturnTypeTuple results; };

}
于 2016-03-07T06:28:23.367 回答
1

非递归方式:

template <typename Func> struct result_of_function;

template <typename Ret, typename ...Args>
struct result_of_function<std::function<Ret(Args...)>>
{
    using type = Ret;
};

template <typename... Tuples> struct tuple_ret_function;

template <typename... Funcs>
struct tuple_ret_function<std::tuple<Funcs...>>
{
    using type = std::tuple<typename result_of_function<Funcs>::type...>;
};

演示

于 2016-03-07T09:25:25.240 回答
1

std::function有一个名为 typedef 的成员result_type。就用它吧。

template<class... Funcs>
auto tuple_function_ret_impl(std::tuple<Funcs...>)
    -> std::tuple<typename Funcs::result_type ...>;

template<class Tuple>
using tuple_function_ret = decltype(tuple_function_ret_impl(Tuple()));

演示

using func_tuple = std::tuple<std::function<int(int)>,
                              std::function<std::string(double)>>;

using ret_tuple = tuple_function_ret<func_tuple>;
using ret_tuple = std::tuple<int, std::string>; // OK
于 2016-03-07T09:52:12.300 回答