Recently I was studying the exact meaning of the well-known "two-phase name lookup" for the names in template classes. Although I have read a lot of articles about this, I still cannot know everything about this. Now I was confusing about the code shown below:
template<typename T>
class A
{
public:
void f(T, T){};
};
namespace ns
{
typedef int TT;
void f(int, int){};
};
template<typename T>
class B : public A<T>
{
public:
void g()
{
//f(T(), T()); // it's fine for error here
typedef ns::TT TTT;
f(TTT(), T()); // why this issued an error?
f(ns::TT(), T()); // and this?
}
};
/* I also think it's OK to move ns here */
// namespace ns
// {
// typedef int TT;
// void f(int, int){};
//};
int main()
{
B<int> b;
b.g();
}
Please notice the second comment. Since "f" is a dependent name, its lookup should be delayed until the instantiation in the "main" function. And at that time, the compiler should perform an argument dependent name lookup at the scope of the main function. I think now it should discover the function in namespace ns, but it still issued a compile error:
1.cpp: In instantiation of 'void B<T>::g() [with T = int]':
1.cpp:30:6: required from here
1.cpp:23:15: error: 'f' was not declared in this scope, and no declarations were found by argument-dependent lookup at the point of instantiation [-fpermissive] f(TTT(), T()); //why this issued an error?
^
1.cpp:23:15: note: declarations in dependent base 'A<int>' are not found by unqualified lookup
1.cpp:23:15: note: use 'this->f' instead
Could someone explain this to me? Thanks.
Argument-dependent lookup only searches the associated classes and namespaces of the argument type. A typedef is just a transparent alias, similarly a using-declaration.
From the draft Standard n3485, [basic.lookup.argdep]/2 about argument-dependent lookup:
[emphasis mine]
As the base class is dependent, it won't be searched during unqualified lookup. Therefore,
f
can be found using argument-dependent lookup. You correctly stated thatf
will be searched for only during the "second phase" (at the point of instantiation), as in your invocations, at least one argument is dependent on a template-parameter.However,
ns
is not a dependent name, as well asTT
(inns::TT
). Therefore, the namespace andTT
must be declared before they're used in the definition ofg
.Whether you write
f(ns::TT(), T())
orf(T(), T())
does not influence the general rule wheref
is searched during argument-dependent lookup: Only the associated namespaces and classes of the types of the arguments (ofT
andns::TT
). Both areint
s forB<int>::g()
, so there are no associated classes and namespaces.f(TTT(), TTT())
changes lookup insofar asf
now is lookup up in the first name lookup phase (it's not a dependent name).Here's an example of argument-dependent lookup:
Now, you can do this as well inside a member function of a class template:
But, as I said, argument-dependent name lookup doesn't work for fundamental types such as
int
.In the example above,
f
is again dependent, as at least one of the arguments is dependent on a template-parameter. Therefore, you could write as well: