Re: Overload lookup of pointer-to-member
On Jul 13, 5:44 am, "jkn...@gmail.com" <jkn...@gmail.com> wrote:
On Jul 12, 3:24 pm, James Kanze <james.ka...@gmail.com> wrote:
On Jul 12, 9:49 am, Abhishek Padmanabh
<abhishek.padman...@gmail.com> wrote:
On Jul 12, 8:14 am, "jkn...@gmail.com" <jkn...@gmail.com> wrote:
[...]
#include <iostream>
using namespace std;
class A
{
public:
int foo()
{
cout << "int foo" << endl;
return 5;
}
template <class A>
void foo (A a)
{
cout << "template foo" << endl;
}
};
template <class Type, class Obj>
void bar (Obj& obj, Type (Obj::*func)())
{
Type t = (obj.*func)();
cout << "TYPE: " << t << endl;
}
int main ()
{
A a;
bar<int>(a, &A::foo); //<int> required, even though only on=
e of the
functions takes no arguments
bar(a, &A::foo); //doesn't compile
}
Compiles fine with Comeau online and VC++ 2005. So, could
be there's something wrong with g++ (an older version, may
be).
I don't know. The problem isn't overload resolution per se,
but template type deduction, which has a completely
different set of rules. In this case, I'll admit that I
don't understand the rules enough to be sure, but off hand,
I don't think it is supposed to work. The compiler can't
effectively do the type deduction for bar until it has
resolved the overloading of &A::foo, and it can't resolve
this overloading until it knows the type &A::foo is being
used to initialize.
In this case, though, it shouldn't necessarily need to know
the return type to determine the proper overloaded function as
it can use the number of arguments.
It can't even start overload resolution until it has done type
deduction. Until the compiler has deduced the type of bar, it
can't consider what instances of foo are value. And until knows
the type of &A::foo, it can't deduce the type of bar.
I'm just a bit confused about the order in which the two
actions are applied (the type deduction and the overload
resolution) and the rules/precedence for resolving overloads
when function templates are involved.
The complete rules are far from simple, but the basic principle
is that the compiler first constructs a set of all possible
overloaded, then does type deduction on the templates, replacing
each template in the initial set with an actual instantiation
(or dropping it completely---substitution failure is not an
error), and only then procedes to overload resolution (the first
step of which is to throw out any functions which cannot be
called. In this particular case, there are two overload
resolutions involved: one for bar and one for foo, but that
doesn't change the basic principles. In order to procede with
the process for &A::foo, it must know the exact target type.
And in order to know the target type, it must have successfully
resolved the overload of bar, which means that it has to have
successfully done type deduction for the template bar. Which
requires knowing which overload of &A::foo was chosen.
As an intelligent human, you can easily solve the problem, but
if you think about it, in solving it, you jump back and forth,
first elimiating some of the overloads for foo (without having
all of the necessary information for full overload resolution),
using intuitive information about the possible results. The
compiler can only do one thing after the other, and in this
case, you have a circular dependency for it: to do A, it must
first do B, and to do B, it must first do A. It doesn't have
enough imagination to be able to speculatively do A and B in
parallel.
--
James Kanze (GABI Software) email:james.kanze@gmail.com
Conseils en informatique orient=E9e objet/
Beratung in objektorientierter Datenverarbeitung
9 place S=E9mard, 78210 St.-Cyr-l'=C9cole, France, +33 (0)1 30 23 00 34