当我们使用构造函数创建对象之后,程序负责跟踪对象,直到对象过期位置。
对象过期时,程序会自动调用一个特殊的成员函数,这个成员函数就叫做析构函数。析构函数这个翻译有一些隐晦,它的英文是 deconstructor,我个人感觉翻译成销毁函数更确切一些。
也就是说当对象不再使用,即将被销毁的时候会调用析构函数。如果我们构造函数当中创建的都是存储持续性的变量,也就是不是使用new
创建的对象。这些对象会自动销毁,并不需要析构函数执行什么逻辑。如果构造函数当中使用了new
动态分配了内存,那么需要在析构函数当中添加对应的delete
语句,将内存释放。
比如这个例子:
class Algo {
public:
Algo(const char* name) {
name_ = new char[strlen(name)+1];
strcpy(name_, name);
}
private:
char *name_;
}
在这个例子当中,Algo
类当中的name_
变量是通过new
动态分配的,那么当Algo
的实例销毁的时候,需要我们在析构函数当中手动执行delete
的逻辑。
析构函数和构造函数几乎完全一样,只在类名前加上~
。析构函数也可以没有返回值和声明类型,并且析构函数没有参数。
加上析构函数之后,上面的例子是这样的:
class Algo {
public:
Algo(const char* name) {
name_ = new char[strlen(name)+1];
strcpy(name_, name);
}
~Algo() {
delete []name_;
}
private:
char *name_;
}
有一点需要注意,析构函数调用应该是由编译器决定,如果创建的是静态存储类对象,则析构函数在程序结束时自动调用,如果创建的是自动存储类对象,析构函数会在程序执行完代码块时被自动调用。如果是通过 new 创建的,那么则在使用delete
时被调用。一般我们不会手动调用析构函数。
由于类对象过期时析构函数会被自动调用,因此必须有一个析构函数。如果程序员没有提供析构函数,那么编译器将隐式地声明一个默认析构函数。