一、名词解释
1,多态
1)静态多态:a) 函数多态:函数重载,运算符重载
b) 模板多态:函数模板,类模板
2)动态多态:虚函数(切记:只有通过指针或引用才能实现动态多态性。"对象.虚函数“ 和 ”类::虚函数“ 这两种形式不能实现多态)
2,静态类型与动态类型
1) 静态类型:声明对象时,对象被赋予的类型,编译时期确定。
2) 动态类型:指针或引用所指对象的类型。常常是,基类指针指向派生类对象,那么这个指针(或引用)的静态类型是基类指针或引用,其动态类型就是派生类指针或引用。
3,静态绑定与动态绑定
1) 静态绑定:亦或静态联编。在编译程序时就根据调用函数提供的信息,把它所对应的具体函数确定下来(编译时期,把调用函数名与具体函数绑定在一起。)
2) 动态绑定:亦或动态联编。在编译程序时还不能确定函数调用所对应的具体函数,只有在程序运行过程中才能够确定函数调用所对应的具体函数。
巧计:动态绑定只发生在继承类的虚函数上(且这个虚函数无缺省参数),其它一概是静态绑定。
二、实例解析
我们看下面这一个示例。
1 #include <iostream>
2 using namespace std;
3 class A
4 {
5 public:
6 virtual void func(int val = 1) {
7 cout<< "A->" << val << endl;
8 }
9 virtual void test() {
10 func();
11 }
12
13 virtual void func2() {
14 cout << "Base A" << endl;
15 }
16 };
17 class B : public A
18 {
19 public:
20 void func(int val = 0) {
21 cout<<"B->"<<val <<endl;
22 }
23 void func2() {
24 cout << "Base B" << endl;
25 }
26 };
27 int main(int argc ,char* argv[])
28 {
29 cout << "Virtual function with default parameters." << endl;
30 B*p = new B; //p的静态类型和动态类型都是 B*
31 p->test();
32 p->func2();
33
34 cout << endl << "reference work with virtual function." << endl;
35 B objectB;
36 A &objectA = objectB; //objectA的静态类型是 A&, 动态类型是 B&
37 objectA.func();
38 objectA.func2();
39
40 cout << endl << "object work with virtual functions." << endl;
41 A objectA2;
42 objectA2.func();
43 objectA2.func2();
44 return 0;
45 }
1,静态类型,动态类型已在程序中注释
2,程序输出
1)第一输出
这一输出是测试带有缺省参数的虚函数。我们看到,基类虚函数的缺省参数没有被修改。(这一情况,总说纷纭,有的把他归于动态绑定,有的归于静态绑定)。所以,尽量不要定义继承而来的缺省参数。
2)第二输出
用基类引用调用派生类虚函数,实现多态。
3)第三输出
基类对象调用自家成员函数,忽略virtual关键字。
三、构造函数,析构函数不要调用虚函数
我们先看下示例代码。
1 #include "stdio.h"
2 class Base
3 {
4 public:
5 Base()
6 {
7 Init();
8 }
9 virtual void Init()
10 {
11 printf("Base Init
");
12 }
13 void func()
14 {
15 printf("Base func
");
16 }
17 };
18 class Derived: public Base
19 {
20 public:
21 virtual void Init()
22 {
23 printf("Derived Init
");
24 }
25 void func()
26 {
27 printf("Derived func
");
28 }
29 };
30
31 int main()
32 {
33 Derived d;
34 ((Base *)&d)->func();
35
36 return 0;
37 }
程序输出为:
Base Init
Base func
我们从输出中可以看到,此时的虚函数并没有发挥它动态绑定的效果。这是因为,在基类构造的时候,虚函数是非虚,不会走到派生类中,既是采用的静态绑定。显然的是:当我们构造一个子类的对象时,先调用基类的构造函数,构造子类中基类部分,子类还没有构造,还没有初始化,如果在基类的构造中调用虚函数,如果可以的话就是调用一个还没有被初始化的对象,那是很危险的,所以C++中是不可以在构造父类对象部分的时候调用子类的虚函数实现。但是不是说你不可以那么写程序,你这么写,编译器也不会报错。只是你如果这么写的话编译器不会给你调用子类的实现,而是还是调用基类的实现。
在析构函数中也不要调用虚函数。在析构的时候会首先调用子类的析构函数,析构掉对象中的子类部分,然后在调用基类的析构函数析构基类部分,如果在基类的析构函数里面调用虚函数,会导致其调用已经析构了的子类对象里面的函数,这是非常危险的。