Как можно определить определение функции шаблона на основе соответствия пакета параметров шаблона аргументам функции? - PullRequest
2 голосов
/ 06 мая 2019

Давайте посмотрим, что у меня есть 2 функции:

int foo(const int, const float);
int bar(const int, const char);

Теперь я хочу перегрузить функцию стандартного шаблона в зависимости от того, соответствует ли она одной из этих функций. Так, например:

template <typename... T>
decltype(foo(declval<T>()...) func(T... args);

template <typename... T>
decltype(bar(declval<T>()...) func(T... args);

Но я получаю ошибку:

ошибка C2995: «функция неизвестного типа (T ...)»: шаблон функции уже определен

Поскольку T должен быть определен по-разному для каждого, я бы предположил, что это допустимая перегрузка, но похоже, что это не так :( Может кто-нибудь помочь мне разрешить эту перегрузку?

1 Ответ

2 голосов
/ 06 мая 2019

Предположим, вы звоните func(0,0).Во время разрешения перегрузки учитываются оба из них:

template <typename... T>
decltype(foo(declval<T>()...) func(T... args);

template <typename... T>
decltype(bar(declval<T>()...) func(T... args);

выполняется подстановка:

template <typename... T = {int, int}>
decltype(foo(declval<int>(),declval<int>()) func(int,  int);

template <typename... T = {int, int}>
decltype(bar(declval<int>(),declval<int>()) func(int, int);

foo и bar оцениваются вызовы, затем decltype 'd:

template <typename... T = {int, int}>
int func(int,  int);

template <typename... T = {int, int}>
int func(int, int);

обратите внимание, что это идентичные подписи.Компилятор жалуется, что вы не можете этого делать.

То, как вы попали к одинаковым сигнатурам, в некотором смысле не имеет значения.

Вы можете написать признакчто гласит «вы можете позвонить bar с этими аргументами».Предположим, вы делаете это.

template<class...Ts>
constexpr bool can_call_bar_with = /* some expression */;

и

template<class...Ts>
constexpr bool can_call_foo_with = /* some expression */;

Теперь мы можем сделать это:

template <typename... T,
  std::enable_if_t< can_call_foo_with<T...>, bool> = true
>
int func(T... args);

template <typename... T,
  std::enable_if_t< can_call_bar_with<T...> && ! can_call_foo_with<T...>, bool> = true
>
int func(T... args);

и теперь, независимо от того, что T... вы передадитеэто, вы никогда не получите два func;это потому, что я гарантировал, что SFINAE сделает только одну действительную подпись.

Для записи

template<class...Ts>
constexpr bool can_call_bar_with = /* some expression */;

существует is_detected или моя can_apply идиома.См. здесь .


Если вы хотите спросить, "какой из foo и` bar предпочтительнее в разрешении перегрузки ", это другая и более сложная проблема,Там нет общего пути;со списком подписей вы можете сделать это.

// вы бы реализовали это примерно так:

template<class...Ts>
struct types_t {};


template<std::size_t I, class Sig>
struct make_tagged_sig;
template<std::size_t I, class Sig>
using tagged_sig = typename make_tagged_sig<I,Sig>::type;

template<std::size_t I, class...Ts>
struct make_tagged_sig<I, types_t<Ts...>> {
  using type=std::integral_constant<std::size_t,I>(Ts...);
};


template<class Sig>
struct overload_check;

template<class R, class...Args>
struct overload_check<R(Args...)> {
  R test(Args...) const;
};

template<class...Sigs>
struct overload_checker:
  overload_check<Sigs>...
{
  using overload_check<Sigs>::test...;

  template<class...Args>
  constexpr auto operator()( types_t<Args...> ) const {
    return decltype( test( std::declval<Args>()... ) ){};
  }
};

template<class Indexes, class...Sigs>
struct which_overload_helper;
template<class...Sigs>
using which_overload_helper_t = typename which_overload_helper<std::index_sequence_for<Sigs...>, Sigs...>::type;
template<std::size_t...Is, class...Sigs>
struct which_overload_helper<std::index_sequence<Is...>, Sigs...> {
    using type = overload_checker< tagged_sig<Is, Sigs>... >;
};

template<class Args, class...Sigs>
constexpr std::size_t which_overload = which_overload_helper_t<Sigs...>{}( Args{} );

Живой пример .

...