我有一个可调用类型的元组。
std::tuple<std::function<int(int)>, std::function<std::string(double)>> funcs;
我想创建另一个具有每个可调用结果类型的元组。例如,funcs
包含int->int
和double->std::string
我怎样才能创建一个results
依赖于每个元素的元组,funcs
看起来像这样。
std::tuple<int, std::string> results;
#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; };
}
非递归方式:
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...>;
};
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