[comp.lang.c++] abstract class constructor pitfall

dove@rocket.UUCP (Webster Dove) (06/06/89)

The constructor a::a() is booby trapped.  Since the virtual mechanism
does not work during construction, the attempted call to foo() is
undefined.  g++-1.35.0 creates an a.out that IOtraps on this.

// -*- C++ -*-

#include <stream.h>

class a
{
public: 
  virtual void foo() = 0;
  a() { foo(); }
};

class c_a : a
{
public:
  virtual void foo() { cout << "c_a\n"; }
};

main()
{
  c_a y;
}
--
		Dr. Webster Dove
		Computing Systems and Architectures
		Advanced Signal Processing Engineering (ASPEN) Dept.
		Sanders Associates (a Lockheed Company)

ark@alice.UUCP (Andrew Koenig) (06/29/89)

In article <DOVE.89Jun6233905@rocket.UUCP>, dove@rocket.UUCP (Webster Dove) writes:
> The constructor a::a() is booby trapped.  Since the virtual mechanism
> does not work during construction, the attempted call to foo() is
> undefined.

	class a
	{
	public: 
	  virtual void foo() = 0;
	  a() { foo(); }
	};

C++ 2.0 is kind enough to give an error message here.
-- 
				--Andrew Koenig
				  ark@europa.att.com

ontologi@bu-tyng.bu.edu (Michael J. Vilot) (06/30/89)

In article <DOVE.89Jun6233905@rocket.UUCP> dove@rocket.UUCP (Webster Dove) writes:
>The constructor a::a() is booby trapped.  Since the virtual mechanism
>does not work during construction, the attempted call to foo() is
>undefined.
>
If, by "booby trapped" you mean that you have left a trap for an unsuspecting
boob, then OK (are you also the office practical joker?)
If, however, you mean "the language is broken" then I must disagree.
You misdiagnose the problem as the virtual mechanism, whose behavior in this 
case is well-defined (just not what you want).

There are 2 things going on here:
	1.  Definition and call of a "pure virtual" function.
	2.  Implicit call of the base class constructor.

>class a {
>public: 
>  virtual void foo() = 0;
			// this 2.0 construct means any call of this
				// function is illegal.  It should be
				// caught at compile time (if possible)
>  a() { foo(); }
			// By the semantics of pure virtuals, this statement
				// should be flagged as an error
>};
>class c_a : a {
>public:
>  virtual void foo() { cout << "c_a\n"; }
>};
>main()
>{
>  c_a y;
		// c_a::c_a() synthesized.
		// Note that c_a()::c_a() will invoke a::a().
>}
>...  g++-1.35.0 creates an a.out that IOtraps on this.

Well, it _should_ catch the error at compile time.
It appears to be deferring the issue in the case of virtual functions, by
generating code for a pure virtual (a::foo()) which is guaranteed to fail.

I think I understand what you are trying to do:  all derived classes should
call their "foo()" as part of their constructor.  But remember: constructors
are _not_ inherited, and they cannot be made virtual.

The solution is to remove the call to the pure virtual (in fact, you could 
probably eliminate the base class constructor entirely, if there's no other
work to be done).  Put the constructor and its call to the virtual function
where they belong -- in the derived class.

The virtual "foo()" will be inherited in every derived class.  If the derived
class programmer forgets to implement a derived "foo()" then the compiler
will help point out the error (hopefully at compile time).

As you can see, the problem you described in <DOVE.89Feb14122134@rocket.UUCP>
has been addressed by (the proper use of) 2.0's "pure virtual" functions.

Hope this helps,

Mike