Шаблон кандидата игнорируется, поскольку аргумент шаблона не может быть выведен
Что не так со следующим фрагментом кода?
#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.
2 ответа
Аргумент 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
), может быть много окружающих типов, которые являются вложенными, или их не должно быть.
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
контекст.
№ 3337 14.8.2.5/4
Однако в определенных контекстах значение не участвует в выводе типа, а вместо этого использует значения аргументов шаблона, которые были либо выведены в другом месте, либо явно указаны. Если параметр шаблона используется только в не выводимых контекстах и не указан явно, вывод аргумента шаблона завершается неудачно.
N3337 14.8.2.5/5
Неведуемые контексты:
- Спецификатор вложенного имени типа, указанного с помощью квалифицированного идентификатора.
Во-первых, используя ::
для ссылки на вложенную структуру неверно. Правильный способ - использоватьA<K>.x
, где x
является членом типа X<K>
. Но чтобы это сработало, вам нужно объявить два членаx
, y
типа X<K>
а также Y<K>
соответственно.
Во-вторых, я думаю, что хорошей практикой является разделение шаблонных объявлений struct X
а также struct Y
из объявления структуры A, чтобы избежать объявления вложенной структуры.
Короче говоря, я бы переписал ваш код следующим образом:
template<class K>
struct X {
K p;
};
template<class K>
struct Y {
K q;
};
template<class K>
struct A {
X<K> x;
Y<K> y;
};
template<class K>
void foo(const X<K>& x, const Y<K>& y) {
std::cout << "A" << std::endl;
}
int main() {
A<float> a;
foo(a.x, a.y);
return 0;
}