0

我创建了一个类,根据 Andrew Koenig 和 Barbara E. Moo 的教科书 Accelerated C++,

析构函数的工作是在对象消失时进行任何清理。通常,此清理涉及释放构造函数已分配的资源,例如内存。

我正在尝试编写一个析构函数,但我对所有漂浮在那里的代码感到困惑。有时使用像这样的简单解构器~MyIntArray() {},有时在{}.

将东西放在大括号之间的规则是什么?是否只是需要放在大括号之间的容器,例如列表、数组、向量、指针(这些是我在代码示例中看到的内容)。

编辑:这是我的课,以防万一

class msgInfo
{
public:
    msgInfo();
    msgInfo(int, int, int, std::string, std::list<int>);

private:
    int source_id;
    int dest_id;
    int priority;
    std::string payload;
    std::list<int> nodePath;
};
4

8 回答 8

3

规则1:

C++03中的三规则或 C++ 11 中的五规则。

如果您的类需要用户定义的复制构造函数或复制赋值运算符,那么它很可能需要用户定义的析构函数。

你什么时候需要这三个?

  • 当您的类具有动态分配的指针成员并且您需要维护每个独立于另一个实例成员的生命周期时。例如:char *会员。
  • 当您管理资源时。例如:打开文件句柄、互斥锁等。

规则 2:

如果您的类打算用于派生并且您需要多态删除对象,那么您必须将 Base 类中的析构函数标记为virtual.

于 2013-03-27T07:38:51.370 回答
2

好吧,如果您动态分配资源(新等),那么在析构函数中您想要释放它们(删除),在您的情况下,由于您的所有成员都不是动态分配的,因此您的析构函数可以为空(或非存在)。

另一个值得一提的注意事项是,如果您最终实现了析构函数,并且您计划让某人继承您的类,则应该将其设为虚拟。

于 2013-03-27T07:38:22.187 回答
1

即使没有明确需要,在 C++ 程序中提供析构函数也是一种很好的编程实践。在您的代码中,您可能没有任何动态内存分配,因此提供的析构函数 ~MyIntArray() {}内部根本没有任何代码。

另请阅读有关 C++ 中的三法则的 Wikipedia 文章。

http://en.wikipedia.org/wiki/Rule_of_three_(C%2B%2B_programming)

于 2013-03-27T07:38:24.373 回答
1

您的类没有任何需要在析构函数中处理的资源:每种类型都是内置的(`int) or handles its own resources (std::string ,std::list`)。所以你不需要实现你自己的析构函数。编译器将提供一个等效于空大括号的一个。

如果你的类有需要处理的资源,你需要自己实现:动态分配的对象、句柄或到套接字的连接、数据库、引用计数等。

实现空析构函数可能有意义的一种情况是,当您有一个旨在派生自并以多态方式使用的类时。在这种情况下,需要一个虚拟析构函数(有很多关于此的 SO 帖子),并且通常的做法是提供一个空实现,以便派生类型在没有资源需要处理时不必自己实现它。

virtual ~Foo() {}
于 2013-03-27T07:38:34.707 回答
1

如果您不提供析构函数,编译器将为您提供一个。这个自动生成的析构函数将正确调用你类的所有数据成员的析构函数,例如payload等。

如果您不需要做任何事情,那么您不需要显式提供析构函数。或者,一个空的也同样有效。

另一方面,如果您的构造函数分配了一些资源(例如,连接到数据库),那么通常您需要在析构函数中放入一些代码来释放该资源(例如,断开与数据库的连接)。这是用于防止资源泄漏的标准 C++ 习惯用法。

于 2013-03-27T07:38:42.150 回答
0

在您的代码中,您可能没有任何动态内存分配,因此您不需要提供析构函数。

于 2013-03-27T09:02:11.263 回答
0

像这样的类不需要一个重要的析构函数(一个,“大括号之间的东西”)。

在析构函数中,您必须释放资源,您已“手动”分配。例如:

  • 如果你new/new[]在构造函数中有并且你需要在析构函数中释放这个内存
  • 如果您在构造函数中打开了文件,请在析构函数中关闭它
  • 如果您在构造函数中锁定了互斥锁,请在析构函数中将其解锁

诸如此类的事情。

此外,当对象被破坏时,您可能需要实现一些额外的逻辑。取决于你想要做什么。

于 2013-03-27T07:39:46.143 回答
0

就我所知或所关心的而言,析构函数的定义应始终如下所示:

~msgInfo() { /* free stuff */ }

另一方面,构造函数可能如下所示:

msgInfo(): m_var1(0), m_var2("Initialize") { /* further initialization */ }

介于两者之间:{是成员变量初始化。

在析构函数中,您应该取消分配在类中其他位置动态分配的任何内容,因此:符号不好,因为您可能需要对delete对象执行操作。

于 2013-03-27T07:41:00.173 回答