虚函数出现是动态联编的需要。
动态联编又称滞后联编、晚期联编。是因为在程序中出现函数调用时,在编译阶段无法确定调用哪一个函数,只有到了程序的运行阶段才能确定调用哪一个函数。
virtual <数据类型> <函数名>( ) {........}
C++中对虚函数的处理方法:在编译阶段不确定调用哪一个函数,在此处保留所有同名虚函数的入口地址,在程序运行时,根据实参的类型来确定。
Tips:
(1) 虚函数,名字,参数个数,参数类型必须完全相同,否则属于函数重载
(2) virtual 基类中不可缺省,派生类可以省略
(3) 虚函数不可为友元函数,因为友元函数不是类的成员函数
(4) 虚函数不可为内联函数,因为内联函数的调用处理在编译时。
(5) 类的构造 函数不能定义为虚函数,但析构函数可以。因为构造函数构造一个对象的时候,必须知道对象的实际类型,而虚函数行为是在运行期间确定实际类型的。而在构造一个对象时,由于对象还未构造成功。编译器无法知道对象 的实际类型,是该类本身,还是该类的一个派生类,或是更深层次的派生类。尤其构造函数中存在有动态申请空间时,在析构函数释放,虚函数的析构函数可以实现撤销对象的动态性
(6) 虚函数一般执行速度会比较慢,为了动态联编,编译器为每个含有虚函数的对象增加了指向虚函数地址表的指针,通过该指针实现虚函数的间接调用。因此尽量不要使用虚函数。
构造函数之所以不能设置成虚函数,主要有以下的几个原因。下面分别阐述一下。
1.虚函数的作用是什么?是实现部分或默认的功能,而且该功能可以被子类所修改。如果父类的构造函数设置成虚函数,那么子类的构造函数会直接覆盖掉父类的构造函数。而父类的构造函数就失去了一些初始化的功能。这与子类的构造需要先完成父类的构造的流程相违背了。而这个后果会相当严重。
2.虚函数的调用是需要通过“虚函数表”来进行的,而虚函数表也需要在对象实例化之后才能够进行调用。在构造对象的过程中,还没有为“虚函数表”分配内存。所以,这个调用也是违背先实例化后调用的准则。
3.虚函数的调用是由父类指针进行完成的,而对象的构造则是由编译器完成的,由于在创建一个对象的过程中,涉及到资源的创建,类型的确定,而这些是无法在运行过程中确定的,需要在编译的过程中就确定下来。而多态是在运行过程中体现出来的,所以是不能够通过虚函数来创建构造函数的,与实例化的次序不同也有关系。
纯虚函数———virtual <数据类型> <函数名>( ) = 0
不等价于virtual <数据类型> <函数名>( ) { }
语法上讲,纯虚函数可以给出函数体。比如可以cout一句话。
定义了纯虚函数的类是抽象类,抽象类只能做基类,不可定义抽象类的对象。
———summerized by Coconut