虚函数的作用,为什么构造函数不能是虚函数

重庆烟草2023-04-25  20

首先,让我们假设他是虚的

当我们在构造函数中时并调用虚函数大家都知道,对于普通的成员函数虚函数的调用是在运行时决定的(即晚捆绑因为在编译时无法知道这个对象是属于这个成员函数的那个类,还是属于由他派生出来的类)

然而,在构造函数中调用虚函数时,他所调用的仅仅是本地版本也就是说,虚函数在构造函数中并不工作!

第一,在概念上,构造函数的工作是把对象变成存在物。在任何构造函数中,对象可能只是部分被形成—我们只能知道基类已被初始化了,但不知道哪个类是从这个基类继承来的。然而,虚函数是“向前”和“向外”进行调用。它能调用在派生类中的函数。如果我们在构造函数中也这样做,那么我们所调用的函数可能操作还没有被初始化的成员,这将导致灾难的发生。

第二,。当一个构造函数被调用时,它做的首要的事情之一是初始化它的VPTR。因此,它只能知道它是“当前”类的,而完全忽视这个对象后面是否还有继承者。当编译器为这个构造函数产生代码时,它是为这个类的构造函数产生代码--既不是为基类,也不是为它的派生类(因为类不知道谁继承它)。

所以它使用的VPTR必须是对于这个类的VTABLE。而且,只要它是最后的构造函数调用,那么在这个对象的生命期内,VPTR将保持被初始化为指向这个VTABLE。但如果接着还有一个更晚派生的构造函数被调用,这个构造函数又将设置VPTR指向它的VTABLE,等直到最后的构造函数结束。VPTR的状态是由被最后调用的构造函数确定的。这就是为什么构造函数调用是从基类到更加派生类顺序的另一个理由。

但是,当这一系列构造函数调用正发生时,每个构造函数都已经设置VPTR指向它自己的VTABLE。如果函数调用使用虚机制,它将只产生通过它自己的VTABLE的调用,而不是最后的VTABLE(所有构造函数被调用后才会有最后的VTABLE)。

另外,许多编译器认识到,如果在构造函数中进行虚函数调用,应该使用早捆绑,因为它们知道晚捆绑将只对本地函数产生调用。无论哪种情况,在构造函数中调用虚函数都没有结果。

所以,构造函数不能是虚的,然而,对于析构函数来说他常常是,而且最好是虚的!这个此处暂时不议

虚函数和实函数的区别如下:

1、虚函数和纯虚函数Q可以定义在同一个类(class)中,含有纯虚函数的类被称为抽象类(abstractclass),而只含有虚函数的类(class)不能被称为抽象类(abstractclass)。

2、虚函数可以被直接使用,也可以被子类(subclass)重载以后以多态的形式调用,而纯虚函数必须在子类(subclass)中实现该函数才可以使用,因为纯虚函数在基类(baseclass)只有声明而没有定义。

3、虚函数和纯虚函数都可以在子类(subclass)中被重载,以多态的形式被调用。

4、虚函数和纯虚函数通常存在于抽象基类(abstractbaseclass-ABC)之中,被继承的子类重载,目的是提供一个统一的接口。

5、对于虚函数来说,父类和子类都有各自的版本。由多态方式调用的时候动态绑定。

1)virtual指明一成员函数为虚函数,而virtual仅用于类的定义里,在类外可不加此关键字

2)一个类的成员函数被定义为虚函数时,子类该函数仍保持虚函数特征

3)子类覆盖此函数时,定义里可不加virtual关键字,但函数声明要和基类的完全一致!且此声明是必须的

4)不是纯虚函数时,父类的虚函数必须要实现; 而若将父类的虚函数设定为纯虚函数时,子类必需要覆盖之而且必须要实现之!

以上就是关于虚函数的作用,为什么构造函数不能是虚函数全部的内容,包括:虚函数的作用,为什么构造函数不能是虚函数、虚函数和实函数的区别、虚函数的用法有哪些等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

转载请注明原文地址:https://juke.outofmemory.cn/read/3672456.html

最新回复(0)