I have the following classes:
class A {
public:
virtual void f() {}
};
class B : public A{
public:
void f(int x) {}
};
If I say
B *b = new B();
b->f();
the compiler says error C2660: 'B::f' : function does not take 0 arguments. Shouldn't the function in B overload it, since it is a virtual function? Do virtual functions get hidden like this?
EDIT: I indeed meant to inherit B from A, which shows the same behaviour.
No, and yes, respectively. If you want the overloading behaviour, you need to say
in B.
When the compiler has more than one way to resolve a symbol, it has to choose which one has precedence unless the code tells it otherwise. What you are expecting is the overloading to take precedence over the overriding. (over, over, over, aaaaack! Sorry, got 'over'whelmed).
This example has B inheriting a virtual method in which the subclass provides an overloaded version. Overloads are for methods in the same class using the same method name but different signatures. Since B is a subclass of A, it is overriding f(), which means it cannot also be an overload at the same time. This is why it is being hidden.
For class A, declaring method
as virtual means that method will be resolved using a certain set of rules that are not consistent with your declaration of b.
By creating 'b' as an instance of "B", the compiler has no need to use the virtual nature of the method of the same name in "A".
If you had declared 'b' like this
then the call b->f(); would indeed refer to the method in A by making use of the virtual resolution.
B does not derive from A, the correct declaration is:
Class B does not derive from A so no function F() exists. You probably meant:
Edit: I missed the actual function hiding. See Steve Jessop answer for more thorough explanation.
Assuming you intended
B
to derive fromA
:f(int)
andf()
are different signatures, hence different functions.You can override a virtual function with a function that has a compatible signature, which means either an identical signature, or one in which the return type is "more specific" (this is covariance).
Otherwise, your derived class function hides the virtual function, just like any other case where a derived class declares functions with the same name as base class functions. You can put
using A::f;
in class B to unhide the nameAlternatively you can call it as
(static_cast<A*>(b))->f();
, or asb->A::f();
. The difference is that ifB
actually does overridef()
, then the former calls the override, whereas the latter calls the function inA
regardless.It seems that it is exist rather similar question with answer in Biern Stroustrup's FAQ: http://www.stroustrup.com/bs_faq2.html#overloadderived
As he said:
but if you want