[comp.lang.c++] Virtual function problem.

mcf@tardis.trl.OZ.AU (Michael Flower) (05/08/91)

I have come across something that appears strange.

If I compile the following with the AT&T 2.1 C++ compiler:

class A {
	virtual void	fn(char);	// 1
	virtual void	fn(char *);	// 2
};

class B : public A {
	void	fn(int);
};

main() {}

in the following way, I get.

"t.c", line 7: warning:  B::fn() hides virtual A::fn()
"t.c", line 7: warning:  B::fn() hides virtual A::fn()

Of course B::fn() hides A::fn(). That is why I wrote it! After all A::fn() is
virtual.

If I change the order of lines 1 and 2 (in class A), the problem goes away.
Admittedly it is only a warning, but I don't like warnings, they often spell
trouble. Is this an anomally?

Michael Flower
Artificial Intelligence Systems         Email:     m.flower@trl.oz.au
Telecom Research Laboratories           Voice:     +61 3 541 6179
Melbourne, AUSTRALIA                    Fax:       +61 3 543 8863

steve@taumet.com (Stephen Clamage) (05/08/91)

mcf@tardis.trl.OZ.AU (Michael Flower) writes:

>class A {
>	virtual void	fn(char);	// 1
>	virtual void	fn(char *);	// 2
>};
>class B : public A {
>	void	fn(int);
>};

>"t.c", line 7: warning:  B::fn() hides virtual A::fn()
>"t.c", line 7: warning:  B::fn() hides virtual A::fn()

>Of course B::fn() hides A::fn(). That is why I wrote it! After all A::fn() is
>virtual.
>If I change the order of lines 1 and 2 (in class A), the problem goes away.

This is a compiler bug, but not in the way you think.  You should always
get the warning.  B::fn(int) hides (not overrides) both A::fn(char) and
A::fn(char*) in that neither function is implicitly accessible from a B
object.  A virtual function in a derived class overrides (not hides) one
in a base class if the signatures are the same (and the return types
must match as well).

In what you show, B::fn(int) is not virtual.  So with your code we can
get the following:
	B b;
	b.fn(2);	// calls B::fn(int)
	b.fn('c');	// calls B::fn((int)'c'), not A::fn('c')
	b.fn("hello");	// illegal
But you could still do this:
	b.A::fn('c');		// calls A::fn(char)
	b.A::fn("hello");	// calls A::fn(char*)

If you declare B::f(int) explicitly virtual, you will get:
	b.fn(2);	// calls B::fn(int)
	b.fn('c');	// calls A::fn(char)
	b.fn("hello");	// calls A::fn(char*)
-- 

Steve Clamage, TauMetric Corp, steve@taumet.com

sdm@cs.brown.edu (Scott Meyers) (05/09/91)

In article <718@taumet.com> steve@taumet.com (Stephen Clamage) writes:
| mcf@tardis.trl.OZ.AU (Michael Flower) writes:
| 
| >class A {
| >	virtual void	fn(char);	// 1
| >	virtual void	fn(char *);	// 2
| >};
| >class B : public A {
| >	void	fn(int);
| >};

...

|          A virtual function in a derived class overrides (not hides) one
| in a base class if the signatures are the same (and the return types
| must match as well).

...

| If you declare B::f(int) explicitly virtual, you will get:
| 	b.fn(2);	// calls B::fn(int)
| 	b.fn('c');	// calls A::fn(char)
| 	b.fn("hello");	// calls A::fn(char*)

Nope, even when B::fn is virtual, it still hides the two versions of A::fn:
  	b.fn(2);	    // calls B::fn(int)
  	b.fn('c');	    // calls B::fn(int)
  	b.fn("hello");	// illegal
Remember that overloaded functions are disambiguated statically, and the
scope of B hides the scope of A.  Hence all references to the identifier fn
through an object of (static) type B will resolve to the fn in B's scope.

Scott


-------------------------------------------------------------------------------
What do you say to a convicted felon in Providence?  "Hello, Mr. Mayor."

bobj@gli.com (Robert Jacobs) (05/09/91)

Michael Flower writes.
> If I compile the following with the AT&T 2.1 C++ compiler:
> 
> class A {
>         virtual void    fn(char);       // 1
>         virtual void    fn(char *);     // 2
> };
> 
> class B : public A {
>         void    fn(int);
> };
> 
> main() {}
> 
> in the following way, I get.
> 
> "t.c", line 7: warning:  B::fn() hides virtual A::fn()
> "t.c", line 7: warning:  B::fn() hides virtual A::fn()
> 
> Of course B::fn() hides A::fn(). That is why I wrote it! After all A::fn() is
> virtual.
> .....

In the ARM on page 210 Stroustrup writes:

	C++ requires an exact type match between a virtual function in a
	base class and a function overriding it in a derived class.

In other words, for a function to be virtual in respect to its base class,
it must match in:

1)	return type
2)	argument types

int != char != char*

Hence the reason for the warning.

fn(int) is not a virtual function.

--------------------------------------------------------------------------
Robert Jacobs                          
bobj@gli.com

jimad@microsoft.UUCP (Jim ADCOCK) (05/10/91)

In article <3472@trlluna.trl.oz> mcf@tardis.trl.OZ.AU (Michael Flower) writes:
|I have come across something that appears strange.
|
|class A {
|	virtual void	fn(char);	// 1
|	virtual void	fn(char *);	// 2
|};
|
|class B : public A {
|	void	fn(int);
|};
|
|main() {}
|
|in the following way, I get.
|
|"t.c", line 7: warning:  B::fn() hides virtual A::fn()
|"t.c", line 7: warning:  B::fn() hides virtual A::fn()
|
|Of course B::fn() hides A::fn(). That is why I wrote it! After all A::fn() is
|virtual.
|
|If I change the order of lines 1 and 2 (in class A), the problem goes away.
|Admittedly it is only a warning, but I don't like warnings, they often spell
|trouble. Is this an anomally?

I think a C++ compiler *ought* to emit a warning when someone overrides
a virtual function with a non-virtual function in a derived class with
a public base.  Do you realize that B::fn(int) is non-virtual?

Maybe the warning ought to be:

warning: B::fn() only *partially* hides virtual A::fn()

[The order you declare things in A shouldn't matter in this issue --
if a compiler is going to issue a warning it should apply no matter
what the ordering of declarations in class A.]

Here's why I think you deserve [at least] a warning for what you've done:

In class A you've declared a virtual fn(char) that you're stating is
part of a polymorphic interface.  When you've publicly inherited A in
class B, you're stating a B can be used as either an A or a B.  If
something is polymorphic we expect its behavior to be the same when 
either accessed as a B or an A:

	B* bp = new B;
	A* ap = bp;
	bp->fn(5);
	ap->fn(5);

And we'd expect the same behavior of fn(5) in either case -- because
fn(5) *appears* to be a virtual function, by virtue of fn(?) being
declared virtual in the base class.  And bp and ap are just alternate
names for the same object -- the object should interpret fn(5) message the
same in either case.  But in reality, the above is resolved as:

	bp->B::fn(5); // B's non-virtual fn(int)
	ap->A::fn(5); // A's virtual fn(char)

so that the behavior of the B object depends on the type of pointer thru
which fn(?) is invoked -- in spite of the fact that in A all flavors of
fn(?) are declared virtual.  One hardly expects this from a "polymorphic"
class, so I think the warning is deserved.

Consider the plight, for example, of a C++ programmer coming from
the C world who hasn't complete absorbed the notion that char and int really 
are really distinct in C++, so that they accidentally declare B::fn(int)
in the derived class, when they really intended to override the virtual
method A::fn(char).  Without the warning [or maybe even with it] they
might spend a long, long time figuring out why they keep getting 
A::fn(char) when invoked as ap->fn(5) -- or alternatively why they 
keep getting B::fn(int) when invoked as bp->fn('A') !!!

If in turn a class C is derived from class B, and some of the fn(?) 
are virtually/non-virtually overridden in C, then the situation becomes
even worse.

One proposal before the ANSI C++ committee would allow the use of
~virtual to allow a programmer to explicitly state that a method was
intended to be non-virtual.  In which case in B you could state:

	~virtual fn(int);	// Don't you warn me, I "know" what I'm doing!

----

I'd recommend that programmers not use functions overloaded by argument
with polymorphic classes -- effectively you're doing double dispatch,
but with the object type resolved dynamically, while the parameter types
resolve statically, which leads to some pretty wierd combinations.  
But, if you insist on doing this, I think they better all be virtual, 
all be declared in the same base class, and all be re-implemented
in any base class that overrides any one of them.

----

A simplified example of the problem:



extern "C" printf(const char*, ...);

class A
{
  public: virtual void print(char c) { printf("as A prints(%c)\n", (int)c); }
};

class B : public A
{
  public: void print(int i) { printf("as B prints(%d)\n", i); }
};

main()
{
  B b;

  B* bp = &b;
  A* ap = &b;

  ap->print('A');
  bp->print('A');

  bp->print(11);
  ap->print(11);
  
return 0;
}

robert@kohlrabi.tcs.com (Robert Blumen) (05/10/91)

In article <3472@trlluna.trl.oz>, mcf@tardis.trl.OZ.AU (Michael Flower) writes:
|> If I compile the following with the AT&T 2.1 C++ compiler:
|> 
|> class A {
|> 	virtual void	fn(char);	// 1
|> 	virtual void	fn(char *);	// 2
|> };
|> 
|> class B : public A {
|> 	void	fn(int);
|> };
|> 
|> in the following way, I get.
|> 
|> "t.c", line 7: warning:  B::fn() hides virtual A::fn()
|> "t.c", line 7: warning:  B::fn() hides virtual A::fn()
|> 
|> Of course B::fn() hides A::fn(). That is why I wrote it! After all A::fn() 
|> is virtual.

The message means that B::fn hides A::fn() and the compiler is wondering
whether perhaps you want B::fn to override A::fn() instead.  I am not sure
why the message goes away when you reverse 1. and 2.  This may have
something to do with what implicit type conversion gets performed.  If you
want B::fn() to override A::fn() for 1. or 2., you must declare it with
exactly the same prototype.

-----------------------------------------------------------------------------
Robert Blumen                          | rblumen@tcs.com
Senior Software Engineer               | 2121 Allston Way, Berkeley, CA 94704
Teknekron Communications Systems, Inc. | (415) 649-3759