c++ - 谁放狗出来的? - 当 "Is-A"在类层次结构中遇到 "Has-A"时对象销毁

标签 c++

考虑一下:

class FluffyThing
{
public:
  FluffyThing()
  {
    m_pMyFur = new Fur;
  }
  virtual ~FluffyThing();

protected:
  Fur * m_pMyFur;
};

class ClawedFluffyThing : public FluffyThing
{
public:
  ClawedFluffyThing()
    : FluffyThing()
  {
    m_pMyClaws = new Claws;
  }
  virtual ~ClawedFluffyThing();

protected:
  Claws * m_pMyClaws;
};

class ScaryFluffyThing : public ClawedFluffyThing
{
public:
  ScaryFluffyThing()
    : ClawedFluffyThing()
  {
    m_pMyTeeth = new Teeth;
    m_pMyCollar = new SpikedCollar;
  }
  virtual ~ScaryFluffyThing();

protected:
  Teeth * m_pMyTeeth;
  SpikedCollar * m_pMyCollar;
};

希望其中没有太多错误 - 我想你明白了。关键是有 3 个类,它们之间存在 IS-A 关系,每个类也有一个或两个属性,在销毁时间时需要进行一些清理。如果我没有声明虚拟析构函数,编译器会自动为我生成以下内容吗?声明了析构函数并因此被迫实现它们(假设使用了类)是进行销毁的正确的长期方法吗?

FluffyThing::~FluffyThing()
{
  delete m_pMyFur;
}

ClawedFluffyThing::~ClawedFluffyThing()
{
  delete m_pMyClaws;
  FluffyThing::~FluffyThing();
}

ScaryFluffyThing::~ScaryFluffyThing()
{
   delete m_pMyTeeth;
   delete m_pMyCollar;
   ClawedFluffyThing::~ClawedFluffyThing();
}

肯定已经有一个明确的答案了……但我无法根据自己的喜好快速掌握它。

最佳答案

没有。您不需要手动调用基类的析构函数,它们会按照与继承相反的顺序自动调用。除此之外,没关系。

FluffyThing::~FluffyThing()
{
  delete m_pMyFur;
}

ClawedFluffyThing::~ClawedFluffyThing()
{
  delete m_pMyClaws;
} //will call ~FluffyThing

ScaryFluffyThing::~ScaryFluffyThing()
{
   delete m_pMyTeeth;
   delete m_pMyCollar;
}  //will call ~ClawedFluffyThing

当然,如果您使用 RAII(智能指针而不是原始指针),您甚至不需要 delete

If I had not declared the virtual destructors, would the compiler have generated the following for me automatically?

没有。 :) virtual 析构函数用于在您通过基类指针删除派生类实例的情况下实现正确的行为。以下内容:

FluffyThing* p = new ScaryFluffyThing;
delete p;

仅当 FluffyThing 的析构函数是虚拟的时才合法。否则,它是未定义的行为。

关于c++ - 谁放狗出来的? - 当 "Is-A"在类层次结构中遇到 "Has-A"时对象销毁,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11944333/

相关文章:

c++ - 无法在 Windows 上编译 SQLCipher

C++ Winsock P2P

C++可执行链接问题

c++ - 简单的 constexpr 函数无法用 GCC 编译(clang 没问题)

c++ - 有没有办法将标准输出设置为二进制模式?

c++ - QSettings : is there a limit in . ini文件行长度?

c++ - 有 Arduino PID 代码的说明吗?

c++ - 为什么前向声明不适用于类?

c++ - 为什么这个本地递归 C++ lambda 如此慢?

c++ - 有没有办法递归地使用类模板参数推导指南? (图灵完备了吗)