2016-09-05 5 views
2

Каковы правила создания экземпляра шаблона, когда мы передаем (мульти) производный класс функции шаблона, ожидающей базовый класс? Например:Создание экземпляра шаблона с множественным наследованием шаблонов

#include <iostream> 

template <int x> 
struct C {}; 

struct D : C<0>, C<1> {}; 

template <int x> 
void f (const C<x> &y) { std::cout << x << "\n"; } 

int main() 
{ 
    f (D()); 
} 

MSVC 2015 оттисков 0, лязг 3.8 - 1 и GCC 6.2 дает ошибку компилятора (Demo). И даже если вы SFINAE одноразовых все перегрузки, кроме одного, то результат все равно будет отличаться:

#include <iostream> 

template <int x> struct C {}; 

template<> 
struct C<0> { using type = void; }; 

struct D : C<0>, C<1> {}; 

template <int x, typename = typename C<x>::type> 
void f (const C<x> &y) { std::cout << x << "\n"; } 

int main() 
{ 
    f (D()); 
} 

Теперь он собирает только с MSVC, и если вы поменяться C<0> и C<1> только лязг будет компилировать его. Проблема заключается в том, что MSVC пытается слишком рано создавать ошибки первой базы, clang-last и gcc-prints. Какой компилятор прав?

+0

Мне кажется, что они все ошибаются. Разве это не двусмысленный вызов функции? –

+0

* «MSVC печатает 0, clang - 1 и gcc дает ошибку компилятора.» *, Какой MSVC, какой gcc и который clang? –

+0

@PiotrSkotnicki Добавлены номера версий, но все они ведут себя одинаково –

ответ

1

GCC 5.4:

/tmp/gcc-explorer-compiler11685-58-1h67lnf/example.cpp: In function 'int main()': 
13 : error: no matching function for call to 'f(D)' 
f (D()); 
^ 
9 : note: candidate: template<int x> void f(const C<x>&) 
void f (const C<x> &y) { std::cout << x << "\n"; } 
^ 
9 : note: template argument deduction/substitution failed: 
13 : note: 'const C<x>' is an ambiguous base class of 'D' 
f (D()); 
^ 
Compilation failed 

Что мне кажется, что правильный результат, так как С < 0> и С < 1> одинаково специализированы.

Тот же результат для GCC 6.2

лязгом 3.8.1 компилирует его, на мой взгляд это ошибка компилятора.

обновление:

Я не знаю фактическое использование случая, но я был удивительно ли это может работать для вас:

#include <utility> 
#include <iostream> 

template<class T> 
struct has_type 
{ 
    template<class U> static auto test(U*) -> decltype(typename U::type{}, std::true_type()); 
    static auto test(...) -> decltype(std::false_type()); 
    using type = decltype(test((T*)0)); 
    static const auto value = type::value; 
}; 

template <int x> struct C {}; 

template<> 
struct C<0> { using type = int; }; 

template<int...xs> 
struct enumerates_C : C<xs>... 
{ 
}; 

struct D : enumerates_C<0, 1> {}; 

template<int x, std::enable_if_t<has_type<C<x>>::value>* = nullptr> 
void f_impl(const C<x>& y) 
{ 
    std::cout << x << "\n"; 
} 

template<int x, std::enable_if_t<not has_type<C<x>>::value>* = nullptr> 
void f_impl(const C<x>& y) 
{ 
    // do nothing 
} 

template <int...xs> 
void f (const enumerates_C<xs...> &y) 
{ 
    using expand = int[]; 
    void(expand { 0, 
     (f_impl(static_cast<C<xs> const &>(y)),0)... 
    }); 
} 

int main() 
{ 
    f (D()); 
} 

ожидается выход (проверено на яблочном лязге):

0 
+0

В первом примере - да, я думаю, он должен быть неоднозначным, во втором - нет, должна быть включена только перегрузка для C <0>. –

+0

«* должна быть включена только перегрузка для C <0> *« что заставляет вас думать, что компилятор должен попытаться использовать каждый возможный «x»? первый дедукция происходит, что не получается –

+0

@PiotrSkotnicki Я задаю этот вопрос, чтобы узнать, как это сделать. Если вы вручную напишете эти функции, то он будет работать так, как планировалось: https://gist.github.com/telishev/a52483833ae6850df69e1e6953f6b277 –

Смежные вопросы