virtual desctructor on pure abstract base class
Asked Answered
M

2

14

I have

struct IMyInterface
{
   virtual method1() = 0;
   virtual method2() = 0;
};

GCC insists that I have

struct IMyInterface
{
   virtual method1() = 0;
   virtual method2() = 0;
   virtual ~IMyInterface(){};
};

I dont see why. A pure interface is all about the interface (duh). The destructor is part of the internal implementation details of a concrete implementer of the interface; it does not form part of the interface. I understand the whole slicing issue (or at least I think I do)

So my question is - is GCC right to insist on it and if so why?

Mcnew answered 26/7, 2010 at 15:46 Comment(3)
You are talking about destructors, but your code shows a constructor. Which is the question about?Shielashield
Rule 1 of Programming: The Compiler is Always Right. Rule 2 of Programming: If the Compiler is Wrong, Rule 1 Applies.Ebon
You can have a pure virtual destructor (virtual ~IMyInterface() = 0;) but if the linker complains about a missing definition, you can actually still provide implementations for pure virtual functions, ie. virtual ~IMyInterface() = 0 {}.Rail
C
25

According to the C++ spec, yes.

You need to declare the destructor virtual because otherwise, later

    IMyInterface * ptr = getARealOne();
    delete ptr;

won't call the destructor on the derived class (because the destructor isn't in the VTable)

It needs to be non-pure because base class destructors are always called by the sub-class destructor.

To further explain, C++ doesn't have a concept of an interface in the same way that Java or C# do. It's just a convention to use only pure-virtual methods, and think of that as an interface. The other rules about C++ destructors make it need to be non-pure, which breaks the similarity to interfaces in other languages, but those languages didn't exist at the time these rules were made.

Casanova answered 26/7, 2010 at 15:50 Comment(6)
Nitpick - I think you can make it pure virtual if you want to require child classes to implement it. But you still have to provide an implementation for the reasons you stated.Decalescence
I don't think so -- it will be called, so it better have a definition. I haven't tried it, but usually a pure-virtual is implemented by the compiler as putting an error function in the VTable to complain that it was called. Perhaps I could imagine a special case for destructors, but I'm pretty sure that that isn't what the spec says (Not 100% though)Casanova
@Lou Franco: Pure virtual functions can have definitions.Elena
@Lou: "pure" doesn't mean that it doesn't have a definition, just that it must be overridden. The destructor can be pure or not, but must have a definition.Patty
Yes -- sorry, that's right. Not the way he's doing it (at the point of declaration).Casanova
Here's the relevant part of the spec discussion open-std.org/jtc1/sc22/wg21/docs/cwg_active.html#230Casanova
G
3

If you don't declare the virtual d'tor in the base class, deleting objects of derived classes through a pointer to the base class leads to the wrong destructor being called, and thus to undefined behaviour and resource leaking.

struct A {

  virtual ~A() {}

};

struct B : A {

   std::string us_constitution;  
};


B* pb = new B();
A* pa = pb;

delete pa; // without the virtual d'tor in the base class, 'B::us_constitution' would never be freed.
Gipsy answered 26/7, 2010 at 15:51 Comment(1)
Nitpick: B::us_constitution cannot be free'd because it was never new'd. The storage it uses, however, will not be returned to the system without the virtual destructor.Elena

© 2022 - 2024 — McMap. All rights reserved.