I'm trying to use a type member of a class in its child, but I'm failing. In particular, I can't understand why this code does not compile:
template <typename T>
class A {
public:
using t = T;
A() {}
};
template <typename T>
class B: public A<T> {
public:
B() : A<T>() {}
A<T>::t foo(A<T>::t x) {
return x;
}
};
I've also tried A::t and B::t but none of them is working. Maybe I'm missing something on how to access inherited type members. Compiler Error:
error: need ‘typename’ before ‘A<T>::t’ because ‘A<T>’ is a dependent scope
A<T>::t foo(A<T>::t x) {
PS: I know that I could use simply T instead of trying to access A::t in this particular case, but the question is still valid.
Since
A<T>::tdepends on a template parameter, it requires atypename:Because it depends on a template parameter, it must always be qualified (i.e. have
::to the left of it), so just the plaintwouldn't work.typename B::twould also work.typename A::twouldn't work because injected-class-nameAitself (i.e. the shorthand version that doesn't require template arguments) also depends on a template parameter. You could dotypename B::A::t, but it's superfluous.If you put this in your class:
using typename A<T>::t;Then you'll be able to use
tas is. Notably,using typename B::t;doesn't work.If the base class didn't depend on a template parameter (e.g.
template <typename T> class B : public A<int>), thentypenamewouldn't be necessary.In that case, all of the following would work:
t,B::t,A::t,B::A::t, optionally with the template arguments forAand/orBspecified.