Подтвердить что ты не робот

Шаблон кандидата игнорируется, поскольку аргумент шаблона не может быть выведен

Что не так в следующем фрагменте кода?

#include <iostream>

template<typename K>
struct A {
    struct X { K p; };
    struct Y { K q; };
};

template<typename K>
void foo(const typename A<K>::X& x, const typename A<K>::Y& y) {
    std::cout << "A" << std::endl;
}

int main() {
    A<float>::X x;
    A<float>::Y y;
    foo(x, y);  
}

clang дает следующее сообщение об ошибке:

17:2: error: no matching function for call to 'foo'
        foo(x, y);      
        ^~~
10:6: note: candidate template ignored: couldn't infer template argument 'K'
void foo(const typename A<K>::X& x, const typename A<K>::Y& y) {
     ^
1 error generated.
4b9b3361

Ответ 1

Аргумент K в const typename A<K>::X не выводим. В принципе, все, что осталось от ::, не выводимо (если :: разделяет вложенное имя).

Тривиально видеть, почему нет смысла просить об удержании, пропустив этот мысленный эксперимент:

struct A { typedef int type; }
struct B { typedef int type; }

template <typename T> void foo(typename T::type);

foo(5);   // is T == A or T == B ??

Нет однозначного сопоставления типов от вложенных типов: для любого типа (например, int) может существовать множество типов окружения, которые являются вложенными типами, или их не должно быть.

Ответ 2

template<typename K>
void foo(const typename A<K>::X& x, const typename A<K>::Y& y) {
    std::cout << "A" << std::endl;
}

K не может быть выведен, поскольку он находится в non-deduced контексте.

n3337 14.8.2.5/4

В некоторых контекстах, однако, значение не участвует в выводе типа, но вместо этого использует значения аргументов шаблона, которые были либо выведены в другом месте, либо явно указаны. Если параметр шаблона используется только в не выводимых контекстов и явно не указано, вывод аргумента шаблона не работает.

n3337 14.8.2.5/5

Невыводимые контексты:

- спецификатор вложенного имени типа, который был указан с использованием идентификатора с квалификацией.