构造函数、析构函数是否要定义为虚函数?

本文探讨了构造函数和析构函数是否应该作为虚函数的问题。基类构造函数不能为虚函数,因为在构造过程中对象的派生类部分尚未初始化。相反,析构函数通常应声明为虚函数,以确保正确地销毁派生类对象。通过示例展示了非虚析构函数可能导致只析构基类而不析构派生类的情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、基类的构造函数不能为虚函数,因为对象不完整

2、不要在构造函数和析构函数里面调用虚函数,可能会发生未定义的行为

       构造派生类对象时,首先调用基类构造函数初始化对象的基类部分。在执行基类构造函数时,对象的派生类部分是未初始化的。实际上,此时的对象还不是一个派生类对象。

       析构派生类对象时,首先撤销/析构他的派生类部分,然后按照与构造顺序的逆序撤销他的基类部分。

因此,在运行构造函数或者析构函数时,对象都是不完整的。为了适应这种不完整,编译器将对象的类型视为在调用构造/析构函数时发生了变换,即:视对象的类型为当前构造函数/析构函数所在的类的类类型。由此造成的结果是:在基类构造函数或者析构函数中,会将派生类对象当做基类类型对象对待。

       而这样一个结果,会对构造函数、析构函数调用期间调用的虚函数类型的动态绑定对象产生影响,最终的结果是:如果在构造函数或者析构函数中调用虚函数,运行的都将是为构造函数或者析构函数自身类类型定义的虚函数版本。 无论有构造函数、析构函数直接还是间接调用虚函数

3、最好把基类的析构函数声明为虚函数

那么在什么情况下,析构函数要定义为虚函数呢?

看一个例子:

class Person
{
public:
	virtual Person* BuyTickets()
	{
		cout << "买票" << endl;
		return this;
	}
	 ~Person()               
	{
		cout << "~Person" << endl;
	}
};

class Student :public Person
{
public:
	virtual Student* BuyTickets()
	{
		cout << "买票——半价" << endl;
		return this;
	}
	~Student()
	{
		cout << "~Student" << endl;
	}
protected:
	int _num;
};

void Fun(Person& p)
{
	p.BuyTickets();
}
void test()
{
	Person* ptr = new Student;         
	delete ptr;                        

}
int main()
{
	test();

	system("pause");
	return 0;
}

运行结果为:

可知只调用了基类的析构函数,改为虚函数为:

class Person
{
public:
	virtual Person* BuyTickets()
	{
		cout << "买票" << endl;
		return this;
	}
	virtual ~Person()               
	{
		cout << "~Person" << endl;
	}
};

class Student :public Person
{
public:
	virtual Student* BuyTickets()
	{
		cout << "买票——半价" << endl;
		return this;
	}
	~Student()
	{
		cout << "~Student" << endl;
	}
protected:
	int _num;
};

void Fun(Person& p)
{
	p.BuyTickets();
}
void test()
{
	Person* ptr = new Student;         
	delete ptr;                        

}
int main()
{
	test();

	system("pause");
	return 0;
}

运行结果为:


所以为了防止只析构基类而不析构派生类的状况发生,我们最好把析构函数声明为成虚函数。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值