C++·多态

定义

派生类的指针可以赋值给基类指针。

通过基类指针调用基类和派生类中的同名虚函数时:

若指向基类,调用基类的虚函数;

若指向派生类,则用派生类的虚函数。

这种机制叫做多态

(窃以为虚函数就是为了让指向不清时通过所指对象知道要调用哪个。)

通过基类引用同理,也算多态。

使用多态可以增强程序的可扩充性,也方便修改。

多态的实质

父类定义共同接口,子类不同实现。

通过父类以相同的方式操作不同子类的行为。

老爸喊:“冲啊!”

老大骑马冲,老二开车冲。

就这个意思。

语法

函数声明中添加关键字virtual,以表示此是虚函数,写在前后都可以。

static不能是virtual的。

继承下来的虚函数,不用再加virtual,编译器也会认为是virtual的。

public,private之类的访问权限检查是根据指针类型来的,语法检查时不考虑指向。

实现

动态联编

一条函数调用语句在编译时无法确定调用哪个函数,运行到该语句时才确定调用哪个函数。

虚函数表

每个有虚函数的类,都有一个虚函数表,该类的任何对象都放着虚函数表的指针。

虚函数表中列出了该类的虚函数地址。

多态的函数调用语句被编译成一系列根据基类指针所指向的(或基类引用所引用的)对象中存放的虚函数表的地址。

在虚函数表中查找虚函数地址,并调用虚函数。

在构造函数或析构函数中调用虚函数时,是静态联编,就调用自己的。

所以通过基类的指针调用派生类的析构函数时,会调用基类的。

但是删除一个派生类的对象时,应该先调用派生类的析构函数,然后调用基类的析构函数。

解决办法:把基类的析构函数声明为virtual。

不允许构造函数是虚的。

纯虚函数

没有函数体的虚函数

virtual void Print() = 0;//纯虚函数,=0是语法定义。

包含纯虚函数的类叫抽象类。

只能用来派生新类,不能创造对象。

抽象类的指针和引用依然可以指向派生类的对象。

在抽象类的成员函数内可以调用纯虚函数。

但是在析构函数和构造函数内不可以。

一个类从抽象类派生而来,它一定要实现基类中的所有纯虚函数(参数必须一模一样),才能成为非抽象类。

C++中的多态(Polymorphism)是指在父类和子类之间的相互转换,以及在不同对象之间的相互转换。 C++中的多态性有两种:静态多态和动态多态。 1. 静态多态 静态多态是指在编译时就已经确定了函数的调用,也称为编译时多态C++中实现静态多态的方式主要有函数重载和运算符重载。 函数重载是指在同一作用域内定义多个同名函数,但它们的参数列表不同。编译器根据传递给函数的参数类型和数量来确定调用哪个函数。例如: ```c++ void print(int num) { std::cout << "This is an integer: " << num << std::endl; } void print(double num) { std::cout << "This is a double: " << num << std::endl; } int main() { int a = 10; double b = 3.14; print(a); // 调用第一个print函数 print(b); // 调用第二个print函数 } ``` 运算符重载是指对C++中的运算符进行重新定义,使其能够用于自定义的数据类型。例如: ```c++ class Complex { public: Complex(double real, double imag) : m_real(real), m_imag(imag) {} Complex operator+(const Complex& other) const { return Complex(m_real + other.m_real, m_imag + other.m_imag); } private: double m_real; double m_imag; }; int main() { Complex a(1.0, 2.0); Complex b(3.0, 4.0); Complex c = a + b; // 调用Complex类中重载的+运算符 } ``` 2. 动态多态 动态多态是指在运行时根据对象的实际类型来确定调用哪个函数,也称为运行时多态C++中实现动态多态的方式主要有虚函数和纯虚函数。 虚函数是在父类中定义的可以被子类重写的函数,使用virtual关键字声明。当一个对象的指针或引用指向一个子类对象时,调用虚函数时会根据实际的对象类型来确定调用哪个函数。例如: ```c++ class Shape { public: virtual void draw() { std::cout << "Drawing a shape." << std::endl; } }; class Circle : public Shape { public: void draw() override { std::cout << "Drawing a circle." << std::endl; } }; int main() { Shape* shape_ptr = new Circle(); shape_ptr->draw(); // 调用Circle类中重写的draw函数 } ``` 纯虚函数是在父类中定义的没有实现的虚函数,使用纯虚函数声明(如virtual void func() = 0;)。父类中包含纯虚函数的类称为抽象类,抽象类不能被实例化,只能作为基类来派生子类。子类必须实现父类的纯虚函数才能实例化。例如: ```c++ class Shape { public: virtual void draw() = 0; }; class Circle : public Shape { public: void draw() override { std::cout << "Drawing a circle." << std::endl; } }; int main() { Shape* shape_ptr = new Circle(); shape_ptr->draw(); // 调用Circle类中重写的draw函数 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值