The key question is that the friend you announced does not match the previous announcement you indicated. The first expects two template parameters, but the second (friend) that you defined accepts only one. Once this is resolved, everything works:
template <class T>
class C;
template <class U, class T>
void func(C<T>& t);
template <class T>
class C
{
template <class U, class TT>
friend void func(C<TT>& t);
private:
template <class U>
void f()
{
}
};
template <class U, class T>
void func(C<T>& t)
{
t.template f<U>();
}
int main() {
C<int> c;
func<bool>(c);
}
.
. U
T
, , T
U
.