What is wrong with the following piece of code?
#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 gives the following error message:
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.
The argument
Kinconst typename A<K>::Xis not deducible. Basically, everything left of a::is not deducible (if::separates a nested-name).It’s trivial to see why it makes no sense to ask for deduction by running through this thought experiment:
There’s no one-to-one mapping from types to nested types: Given any type (such as
int), there could be many ambient types of which it is a nested type, or there needn’t be any.