多态通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态。例如下图所示,移动支付(微信、支付宝)的抢红包活动,每个人所抢到的红包金额都不尽相同,有运气王也有手气一般的。
多态是在不同继承关系的类对象,去调用同一函数,产生了不同的行为。例如Student继承了Person。Person对象买票全价,Student对象买票半价。
那么在继承中要构成多态还有两个条件:
class Person {
public:
virtual void BuyTicket() {
cout << "买票-全价" << endl;
}
};
class Student :public Person {
public:
virtual void BuyTicket() {
cout << "买票-半价" << endl;
}
};
void Func(Person& people) {
people.BuyTicket();
}
void test_polymorphism01() {
Person p;
Student s;
Func(p);
Func(s);
}
根据图3的调试结果看以:不同的对象(普通人p、学生s)去买票的结果是不一样的,其中普通人票价是全价,学生的票价是半价。
虚函数即被virtual
修饰的类成员函数称为虚函数。例如下面案例的代码所示的虚函数:
class Person {
public:
virtual void BuyTicket() {
cout << "买票-全价" << endl;
}
};
虚函数的重写(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名、参数列表完全相同),称子类的虚函数重写了基类的虚函数。
class Person {
public:
//重写 体现的是接口的继承,重写继承基类这个函数的实现
virtual void BuyTicket() {
cout << "买票-全价" << endl;
}
virtual ~Person() {
cout << "~Person()" << endl;
}
};
class Student :public Person {
public:
//这里的virtual可以不写,因为它继承父类的接口,重写实现
void BuyTicket() {
cout << "买票-半价" << endl;
}
~Student() {
cout << "~Student()" << endl;
}
};
void Func(Person* p) {
p->BuyTicket();
delete p;
}
void test_polymorphism02() {
Func(new Person);
Func(new Student);
}
注意:在重写基类虚函数时,派生类的虚函数在不加virtual关键字时,虽然也可以构成重写(因为继承后基类的虚函数被继承下来了在派生类依旧保持虚函数属性),但是该种写法不是很规范,不建议使用。
虚函数重写的两个例外:
派生类重写基类虚函数时,与基类虚函数返回值类型不同。即基类虚函数返回基类对象的指针或引用,派生类虚函数返回派生类对象的指针或引用时称为协变。
class Person {
public:
virtual Person* BuyTicket() {
cout << "买票-全价" << endl;
return this;
}
};
class Student :public Person {
public:
virtual Student* BuyTicket() {
cout << "买票-半价" << endl;
return this;
}
};
void Func(Person& people) {
people.BuyTicket();
}
void test_polymorphism03() {
Person p;
Student s;
Func(p);
Func(s);
}
具体的调试结果如图5所示:
如果基类的析构函数为虚函数,此时派生类析构函数只要定义,无论是否加virtual
关键字,都与基类的析构函数构成重写,虽然基类与派生类析构函数名不同,看起来违背了重写规则,其实不然,这里可以理解为编译器对析构函数的名称做了特殊处理,编译后析构函数的名称统一处理成destructor
。
class Person {
public:
virtual ~Person() {
cout << "~Person()" << endl;
}
};
class Student :public Person {
public:
virtual ~Student() {
cout << "~Student()" << endl;
}
};
void test_polymorphism04() {
Person* p1 = new Person;
Person* p2 = new Student;
delete p1;
delete p2;
}
具体的调试结果如图6所示:
从上面可以看出,C++对函数重写的要求比较严格,但是有些情况下由于疏忽,可能会导致函数名字母次序写反而无法构成重载,而这种错误在编译期间是不会报出的,只有在程序运行时没有得到预期结果才来debug会得不偿失,因此:C++11提供了override和final两个关键字,可以帮助用户检测是否重写。
class Car {
public:
virtual void Drive() final {
}
};
class Benz :public Car {
public:
virtual void Drive() {
cout << "Benz-舒适" << endl;
}
};
具体的调试结果如图7所示:
class Car {
public:
void Drive() {
}
};
class Benz :public Car {
public:
virtual void Drive() override {
cout << "Benz-舒适" << endl;
}
};
具体的调试结果如图8所示:
依据报错修改后的代码如下:
class Car {
public:
virtual void Drive() {
}
};
class Benz :public Car {
public:
virtual void Drive() override {
cout << "Benz-舒适" << endl;
}
};
在虚函数的后面写上=0,则这个函数为纯虚函数。包含纯虚函数的类叫做抽象类(也叫接口类),抽象类不能实例化出对象。派生类继承后不能实例化出对象,只有重写纯虚函数,派生类才能实例化出对象。纯虚函数规范了派生类必须重写,另外纯虚函数更体现出了接口继承。
class Car {
public:
//纯虚函数
virtual void Drive() = 0;
};
class Benz :public Car {
public:
virtual void Drive() {
cout << "Benz-舒适" << endl;
}
};
class BMW :public Car {
public:
virtual void Drive() {
cout << "BMW-操控" << endl;
}
};
void test_polymorphism07() {
Car c;
}
具体的调试结果详见图10,可知抽象类的对象无法实例化:
//派生类虚函数重写后可以实例化出对象
void test_polymorphism08() {
Benz bz;
BMW bw;
}
注意:一个类型在现实中没有对应的实体,可以将该类定义为抽象类。
普通函数继承是一种实现继承,派生类继承了基类函数,可以使用函数,继承的是函数的实现。虚函数的继承是一种接口继承,派生类继承的是虚函数的接口,目的是为了重写,达成多态,继承的是接口。所以如果不实现多态,不要把函数定义成虚函数。
class Base {
public:
virtual void Func1() {
cout << "Func1()" << endl;
}
private:
int _b = 1;
char _ch;
};
void test_polymorphism08() {
cout << sizeof(Base) << endl;
Base bb;
}
通过调试可知对象bb占用12byte,除了成员变量_b
、_ch
外,还多一个__vfptr放在对象的前面(注意有些平台可能会放到对象的最后面,这个和平台有关),对象中的这个指针我们叫做虚函数表指针(v代表virtual,f代表function)。一个含有虚函数的类中都至少有一个虚函数表指针,因为虚函数的地址要被放到虚函数表中,虚函数表简称虚表。
依然是买票问题,经过分析可知:指针或引用指向基类对象调用基类的虚函数,指向派生类对象则调用派生类的虚函数。基类对象的虚表存放基类的虚函数,派生类对象的虚表存放派生类的虚函数。
class Person {
public:
virtual void BuyTicket() {
cout << "买票-全价" << endl;
}
};
class Student :public Person {
public:
virtual void BuyTicket() {
cout << "买票-半价" << endl;
}
};
void Func(Person& people) {
people.BuyTicket();
}
void test_polymorphism09() {
Person mike;
Student johnson;
Func(mike);
Func(johnson);
}
具体调试解果如图12所示:
people
指向mike
对象时,people->BuyTicket
在mike
的虚表中找到虚函数是Person::BuyTicket
。people
指向johnson
对象时,people->BuyTicket
在johnson
的虚表中找到虚函数是Student::BuyTicket
。class Base {
public:
virtual void Func1() {
cout << "Base::Func1()" << endl;
}
virtual void Func2() {
cout << "Base::Func2()" << endl;
}
void Func3() {
cout << "Base::Func3()" << endl;
}
private:
int _b = 1;
};
class Derive :public Base {
public:
virtual void Func1() {
cout << "Derive::Func1()" << endl;
}
private:
int _d = 2;
};
void test_polymorphism10() {
Base b;
Derive d;
}
派生类Derive对象的虚表里拷贝了基类Base的虚表,并将虚函数Func1()
地址进行重写(覆盖),对于虚表中Func2()
虚函数的地址直接继承了基类Base的虚函数Func2()
的地址。
在Student类中添加虚函数Func4代码,结果在监视窗口查看时发现虚表中未发现Func4的指针。
virtual void Func4() {
cout << "Derive::Func4()" << endl;
}
若不想通过调式时使用监控窗口或内存来查看虚表,同时也为了应对图14的bug,亦可以使用下列代码打印虚表:
//用程序打印虚表
//声明一个函数指针 VF_PTR代表void(*)()
typedef void(*VF_PTR)();
//虚函数表本质是一个存虚函数指针的指针数组,一般情况这个数组最后面放了一个nullptr(VS系列编译器G++没有)
void PrintVFTable(VF_PTR table[]) {
for (int i = 0; table[i] != nullptr; ++i) {
printf("[%d]:%p->", i, table[i]);
VF_PTR f = table[i];
f();
}
cout << endl;
}
void test_polymorphism10() {
Base b;
Derive d;
//能适应32位和64位平台
cout << "b的虚表地址:" << endl;
PrintVFTable(*(VF_PTR**)&b);//&b==Base* -> (int*)&b == (int*)(Base*)
cout << "d的虚表地址:" << endl;
PrintVFTable(*(VF_PTR**)&d);
}
图15可以发现打印结果与调式时内存结果完全相同:
小结:
静态绑定又称前期绑定(早绑定),在程序编译期间确定了程序的行为,也成为静态多态,例如:函数重载;
动态绑定又称后期绑定(晚绑定),是在程序运行期间,根据具体拿到的类型确定程序的具体行为,调用具体的函数,也称为动态多态。
需要注意的是在单继承和多继承关系中,需要关注派生类对象的虚表模型。
详见4.2节多态原理案例Derive对象的虚表无法现实func4()指针。
class Base1 {
public:
virtual void func1() {
cout << "Base1::func1()" << endl;
}
virtual void func2() {
cout << "Base1::func2()" << endl;
}
private:
int b1;
};
class Base2 {
public:
virtual void func1() {
cout << "Base2::func1()" << endl;
}
virtual void func2() {
cout << "Base2::func2()" << endl;
}
private:
int b2;
};
class Derive :public Base1, public Base2 {
public:
virtual void func1() {
cout << "Derive::func1()" << endl;
}
virtual void func3() {
cout << "Derive::func3()" << endl;
}
private:
int d1;
};
typedef void(*VF_PTR)();
void PrintVTable(VF_PTR vTable[]) {
cout << " 虚表地址" << vTable << endl;
for (int i = 0; vTable[i] != nullptr; ++i) {
printf("[%d]:%p->", i, vTable[i]);
VF_PTR f = vTable[i];
f();
}
cout << endl;
}
void test_polymorphism11() {
Derive d;
PrintVTable((VF_PTR*)(*(int*)&d));
//PrintVTable((VF_PTR*)(*(int*)((char*)&d+sizeof(Base1))));
Base2* ptr2 = &d;
PrintVTable((VF_PTR*)(*(int*)(ptr2)));
}
调试结果具体如图所示:
此外,上图可知:多继承后,虚表中重写的func1()的地址不一样。
在实际工程中,不建议设计出菱形继承及菱形虚拟继承,一方面太复杂容易出问题,另一方面这样的模型,访问基类成员有一定性能损耗。所以菱形继承、菱形虚拟继承的虚表我们一般看不到。
class A {
public:
virtual void func1() {
}
public:
int _a;
};
class B :virtual public A {
public:
virtual void func1() {
}
virtual void func2() {
}
public:
int _b;
};
class C :virtual public A {
public:
virtual void func1() {
}
virtual void func3() {
}
public:
int _c;
};
class D :public B, public C {
public:
virtual void func1() {
}
public:
int _d;
};
void test_polymorphism12() {
D d;
d.B::_a = 1;
d.C::_a = 2;
d._b = 3;
d._c = 4;
d._d = 5;
}
菱形虚继承的虚表、虚基表,如图17所示: