跑完你这个程序也没什么呀,有什么问题。不知道你要说什么
#include <stdio.h>
#include <iostream>
using namespace std;
class B
{
public:
B(void){P();}
virtual void P(void){std::cout<<"This is B"<<std::endl;}
};
class D
{
public:
D(void){P();}
virtual void P(void){std::cout<<"This is D"<<std::endl;}
};
编译器正式为了能够在构造函数里面调用正确的虚函数,所以才会在构造函数刚开是,初始化参数列表之前对vptr进行初始化。而不会在构造函数结尾。
#include <iostream>
using namespace std;
class A
{
public :
A()
{
p() ;
}
virtual void p() {cout<<"A:"<<sizeof(A)<<endl;};
};
class B : public A
{
public:
int i;
B():i(10)
{
p();
}
virtual void p(){cout<<"B:"<<sizeof(B)<<endl;} ;
};
int main()
{
B b;
system("pause");
the effect of making a virtual call to a pure virtual function directly or indirectly for the object being created from such a constructor is underfined.
14.8.2 虚函数在构造函数中的行为
构造函数调用层次会导致一个有趣的两难选择。试想;如果我们正在构造函数中并且调用
虚函数,那么会发生什么现象呢?对于普通的成员函数,虚函数的调用是在运行时决定的,这
是因为编译时并不能知道这个对象是属于这个成员函数所在的那个类,还是属于由它派生出来
的类。于是,我们也许会认为在构造函数中也会发生同样的事情。
然而,情况并非如此。对于在构造函数中调用一个虚函数的情况,被调用的只是这个函数
的本地版本。也就是说,虚机制在构造函数中不工作。
这个行为有两个理由。在概念上,构造函数的工作是把对象变成存在物。在任何构造函数
中,对象可能只是部分被形成—我们只能知道基类已被初始化了,但不知道哪个类是从这个
基类继承来的。然而,虚函数是“向前”和“向外” 进行调用。它能调用在派生类中的函数。
如果我们在构造函数中也这样做,那么我们所调用的函数可能操作还没有被初始化的成员,这
将导致灾难的发生。
第二个理由是机械的。当一个构造函数被调用时,它做的首要的事情之一是初始化它的
V P T R。因此,它只能知道它是“当前”类的,而完全忽视这个对象后面是否还有继承者。当
编译器为这个构造函数产生代码时,它是为这个类的构造函数产生代码- -既不是为基类,也不
是为它的派生类(因为类不知道谁继承它)。所以它使用的V P T R必须是对于这个类的V TA B L E。
而且,只要它是最后的构造函数调用,那么在这个对象的生命期内, V P T R将保持被初始化为
指向这个V TA B L E。但如果接着还有一个更晚派生的构造函数被调用,这个构造函数又将设置
V P T R指向它的V TA B L E,等等,直到最后的构造函数结束。V P T R的状态是由被最后调用的构
造函数确定的。这就是为什么构造函数调用是从基类到更加派生类顺序的另一个理由。
但是,当这一系列构造函数调用正发生时,每个构造函数都已经设置V P T R指向它自己的
V TA B L E。如果函数调用使用虚机制,它将只产生通过它自己的V TA B L E的调用,而不是最后
的V TA B L E(所有构造函数被调用后才会有最后的V TA B L E)。另外,许多编译器认识到,如
果在构造函数中进行虚函数调用,应该使用早捆绑,因为它们知道晚捆绑将只对本地函数产生
调用。无论哪种情况,在构造函数中调用虚函数都没有结果