多态(第五次作业)
多态:按字面的意思就是多种形态。当类之间存在层次结构,并且类之间是通过继承关联时,就会用到多态。
C++ 多态意味着调用成员函数时,会根据调用函数的对象的类型来执行不同的函数。
有例子如下:
#include<iostream> using namespace std; class Shape { protected: int width,height; public: Shape(int a = 0,int b = 0) { width = a; height = b; } int area() { cout << "Parent class area : " << endl; return 0; } }; class Rectangle : public Shape { public: Rectangle(int a = 0,int b = 0):Shape(a,b){} int area() { cout << "Rectangle class area : " << endl; return (width*height); } }; class Triangle: public Shape { public: Triangle(int a = 0,int b = 0):Shape(a,b){} int area() { cout << "Triangle class area : " << endl; return (width*height/2); } }; int main() { Shape* shape; Rectangle rec(10,7); Triangle tri(10,5); shape = &rec; shape->area(); shape=&tri; shape->area(); return 0; }
程序运行结果如下:
导致输出错误的原因是调用area()被编译器设置为基类中的版本,这就是所谓的静态多态,或静态链接。函数调用在程序执行前就准备好了。有时候这也被称为早绑定因为area()函数在程序编译期间就已经设置好了。
现在我们对程序稍作修改,在Shape类中的,area()的声明前放置关键字virtual:
class Shape { protected: int width,height; public: Shape(int a = 0,int b = 0) { width = a; height = b; } virtual int area() { cout << "Parent class area : " << endl; return 0; } };
运行结果如下:
此时编译器看到的是指针内容,而不是它的类型。因此由于tri和rec类的对象的地址存储在*shape中,所以会调用各自的area()函数。如上所示,每个子类都有一个函数area()的独立实现,这就是多态的一般使用方法。有了多态就可以有多个不同的类,都带有同一个名称但具有不同实现的函数,函数的参数甚至都是相同的。
虚函数
虚函数是基类中使用关键字virtual声明的函数。在派生类中重新定义基类中的虚函数,会告诉编译器不要静态静态链接到该函数。
我们想要的是在程序中任意点可以根据所调用的对象类型来选择调用的函数。这种操作被称为动态链接,或后期绑定。
纯虚函数
你可能想要在基类中定义虚函数,以便在派生类中重新定义该函数更好的适用于对象,但是你在基类中又不能对虚函数给出有意义的实现,这个时候就会用到纯虚函数。
class Shape { protected: int width,height; public: Shape(int a = 0,int b = 0) { width = a; height = b; } virtual int area() = 0; };
= 0 告诉编译器,函数没有主题,上面的虚函数就是纯虚函数。
- 纯虚函数声明如下:virtual void function1() = 0; 纯虚函数一定没有定义,纯虚函数用来规范派生类的行为,即接口。包含纯虚函数的类是抽象类,抽象类不能定义实例,但可以声明指向实现该抽象类的具体类的指针或引用。
- 虚函数声明如下:virtual ReturnType FunctionName(Parameter) 虚函数必须实现,如果不实现,编译器经报错。
- 对于虚函数来说,父类和子类都有各自的版本,由多态方式调用的时候动态绑定。
- 实现了纯虚函数的子类,该纯虚函数在子类中就编程了虚函数,子类的子类即孙类可以覆盖该虚函数,由该虚函数,由多态方式调用的是动态绑定。
- 虚函数是C++中用来实现多态(polymorphism)的机制。核心理念就是通过基类访问派生类定义的函数
- 在动态分配堆上内存的时候,析构函数必须是虚函数,但没有必要时纯虚函数。
- 友元不是成员函数,只要成员函数才能是虚拟的。因此友元函数不能是虚函数,但可以通过让友元函数调用虚拟成员函数来解决友元函数的虚拟问题。
- 析构函数应当是虚函数,将调用相应对象类型的析构函数,因此,如果指针指向的是子类对象,将调用子类的析构函数,然后自动调用基类的析构函数。