混合void_t和可变参数模板?

时间:2017-12-19 22:31:10

标签: c++ variadic-templates c++17 sfinae void-t

请考虑以下代码:

template <class F, class... Args, class = std::void_t<>>
struct is_invokable
: std::false_type {};
template <class F, class... Args>
struct is_invokable<F, Args..., std::void_t<std::invoke_result_t<F, Args...>>>
: std::true_type {};

目标是让一个特性能够判断F类型的可调用类是否可以使用Args...类型的参数调用。

然而,它无法编译,因为:

error: parameter pack 'Args' must be at the end of the template parameter list

在C ++ 17中执行此操作的(优雅)方式是什么?

2 个答案:

答案 0 :(得分:6)

namespace details {
  template <class F, class, class... Args>
  struct is_invokable : std::false_type {};
  template <class F, class... Args>
  struct is_invokable<F, std::void_t<std::invoke_result_t<F, Args...>>, Args...>
  : std::true_type {};
}
template <class F, class... Args>
using is_invokable=typename ::details::is_invokable<F, void, Args...>::type;

答案 1 :(得分:1)

我建议帮助structis_invokable_h)并使用std::tuple来包裹Args...

这样的东西
#include <type_traits>
#include <utility>

template <typename, typename, typename = void>
struct is_invokable_h : std::false_type
 {};

template <typename F, typename ... Args>
struct is_invokable_h<F, std::tuple<Args...>,
                      std::void_t<std::invoke_result_t<F, Args...>>>
   : std::true_type
 {};

template <typename F, typename ... Args>
struct is_invokable : is_invokable_h<F, std::tuple<Args...>>
 {};

int foo (int)
 { return 0; }

int main()
 {
   static_assert( true  == is_invokable<decltype(foo), int>{} );
   static_assert( false == is_invokable<decltype(foo), int, int>{} );
 }