当你每次看到C++类中声明一个virtual函数,特别是看到了一个virtual的虚构函数。你知道它的意思吗?你肯定会毫不犹豫的回答:不就是多态么。。。在运行时确定具体的行为么。。。完全正确,但这里我要讲的不只是这些东西。
有些类需要虚函数,有些不需要虚函数。这是为什么,一般你看到的类如果有一个虚析构函数,那么这个类中应该会有至少一个是虚函数的。。这是为什么呢??如果我们类中没有用其他虚函数的话,你创建了这个也是多余的,而且会增加类对象的大小。。说这些纯理论的东西,也许大家不知所云。。下面我就给例子来验证。。
1:
class A
{
public:A(){};
// virtual ~A(){};~A();
};
void main()
{A a;cout<<sizeof(a)<<endl;
}
结果为1。这个1应该是编译器自己为它加上的。。哪怕你不在类中不写任何东西,它也是1;例如;
class A
{};
void main()
{A a;cout<<sizeof(a)<<endl;
}
如果你把析构函数声明为虚函数。。如:
2:
class A
{
public:
A(){};
virtual ~A(){};
//~A();
};
void main()
{
A a;
cout<<sizeof(a)<<endl;
}
结果是4。先不说这是为什么。。
然后还是说一下关于虚函数基础的东西(多态)吧,也给个例子:
#include<iostream>
#include<string>
using namespace std;class Base
{
public:Base();virtual ~Base();virtual void test();
private :int count;
};
Base::Base(){cout<<"Base部分创建了"<<endl;
}
Base::~Base(){cout<<"Base部分被销毁了"<<endl;
}
void Base::test()
{cout<<"Base Test"<<endl;}class Derive1:public Base
{
public:Derive1();virtual ~Derive1();void test();};
Derive1::Derive1(){cout<<"子类部分创建了"<<endl;
}
Derive1::~Derive1(){cout<<"子类部分被销毁了"<<endl;
}
void Derive1::test()
{cout<<"Derive1 Test"<<endl;
}void main()
{Base* d1=new Derive1();d1->test();delete d1;}
由此看见,当通过声明一个父类指针并且让它指向一个子类的对象,在子对象创建的时候,会先去调用父类的构造函数,然后再是自己的构造函数,当通过父类指针去调用一个虚函数test()时,它实际上回去调用子类的test()函数,这是为什么呢,它肯定有什么信息让它这样做吗。。这个信息肯定是子类对象给它的。。这个信息就是虚函数指针(vptr),它指向一个虚函数表(vtbl),这个虚函数表其实就是包含了这个类的所有虚函数的函数名(函数指针),每个类就只包含了那一个虚函数指针和它的一些成员变量。这下可以解释上面为什么是1,为什么是4了。。
在win32的机器上,每个指针是4字节。刚才也提到每个类的大小取决于两部分,一个是成员变量,一个是虚函数指针而且有且只有一个,在例子一中,因为没有成员变量,而有一个虚函数---析构函数,此时肯定会有一个虚函数指针,所以是4。。 其实刚才也就同时说清楚了多态的本质,就是子对象的虚函数指针给出了这个信息,父类指针才知道去执行哪个函数。。
最后一个问题:为什么析构函数要声明为虚函数呢?(当至少有一个为虚函数的时候)
从刚才的那个结果也可以看出,当我们delete那个指针的时候,会发生析构,而且这个过程是从子类到父类的顺序进行。假如此时析构函数不为虚函数,父类指针也就不知道去执行子类的析构函数。。也就不会去释放子对象的那部分内存,造成内存泄漏。。例如:(这里我们只是对上一段代码进行修改,去掉了父类中的virtual):
#include<iostream>
#include<string>
using namespace std;class Base
{
public:Base();~Base();virtual void test();
private :int count;
};
Base::Base(){cout<<"Base部分创建了"<<endl;
}
Base::~Base(){cout<<"Base部分被销毁了"<<endl;
}
void Base::test()
{cout<<"Base Test"<<endl;}class Derive1:public Base
{
public:Derive1();~Derive1();void test();};
Derive1::Derive1(){cout<<"子类部分创建了"<<endl;
}
Derive1::~Derive1(){cout<<"子类部分被销毁了"<<endl;
}
void Derive1::test()
{cout<<"Derive1 Test"<<endl;
}void main()
{Base* d1=new Derive1();d1->test();delete d1;}
所以当至少有一个虚函数的话,我们也要把它的析构函数声明为virtual。(插一句:有些人会说你子类中的那个函数哪里是虚函数哦,我没看到virtual 啊。。其实C++允许我们这样做,重写父类的虚函数,不是必须要声明出来的。)
总结:一个类有了虚函数,是为了成为一个基类,如果不是这样的话,那么父类中的任何函数都没有必要是虚函数,甚至会增加类的大小。多态告诉了我们这点。。一旦成为了基类,那么就要把析构函数声明为一个虚函数。。
好了,虚函数的内容就Over了。。。。。