c++多态与虚函数

多态

在通用主流语言中,多态的概念是指把子类对象赋值给父类指针的时候,从而触发父类指针分别调用不同子类对象成员函数的现象。
归纳总结为两点:

  1. 父类指针 指向 子类对象
  2. 根据不同子类对象,父类指针能调用不同的子类对象的成员函数。

虚函数

c++中是用虚函数来实现多态的,虚函数的特征:

  1. 被virtual修饰的成员函数就是虚函数。
  2. 虚函数属性具有向下传递性,下面所有子类只要重写了父类的虚函数,所有子类只要重写了父类虚函数的均默认为虚函数(哪怕不写virtual)。

虚函数的简单用法:

class Animal
{
public:
    int m_age;
    virtual void speak() {
        cout << "Animal:speak()" << endl;
    }
    virtual void run() {
        cout << "Animal:run()" << endl;
    }
};

class Cat : public Animal
{
public:
    int m_life;
    void speak() {
        cout << "Cat:speak()" << endl;
    }
    void run() {
        cout << "Cat:run()" << endl;
    }
};

Animal* cat = new Cat();
cat->m_age = 20;
cat->speak();  //调用 Cat:speak() 
cat->run();  //调用 Cat:run()
delete cat;

虚函数的原理

当编译器检测到类中存在虚函数,则会自动生成一张虚表,虚表记录了类中虚函数的地址,然后把虚表的地址放在子类对象的首地址,这样就能实现用父类指针 调用 子类对象的虚函数了。
所有的子类对象(不管在全局区、栈、堆)共用同一份虚表。

如图所示:


a.png

子类虚函数中调用父类虚函数的方法

class Animal
{
public:
    virtual void speak() {
        cout << "Animal:speak()" << endl;
    }
};

class Cat : public Animal
{
public:
    void speak() {
        Animal::speak();
        cout << "Cat:speak()" << endl;
    }
};

虚析构函数

当父类指针 指向 子类对象的时候,如果没有声明父类析构函数为虚析构函数,则会导致只调用父类的析构函数,而不会调用子类的析构函数,从而导致内存泄露。

因此需要声明父类中的虚析构函数:

class Animal
{
public:
    virtual void speak() {
        cout << "Animal:speak()" << endl;
    }
    virtual ~Animal() {
        cout << "~Animal()" << endl;
    }
};

class Cat : public Animal
{
public:
    void speak() {
        Animal::speak();
        cout << "Cat:speak()" << endl;
    }
    ~Cat() {
        cout << "~Cat()" << endl;
    }
};

纯虚函数与抽象类

纯虚函数是指没有函数体且初始化为0的虚函数,通常用来定义接口规范。
含有纯虚函数的类即为抽象类,抽象类具有以下特性:

  1. 不可实例化
  2. 父类是抽象类,子类没有完全重写纯虚函数,子类依然是抽象类
  3. 抽象类可以拥有成员变量和非纯虚函数

以下为简单用法:

class Animal
{
public:
    int m_age;
    virtual void speak() = 0;
    virtual void run() = 0;
    void see() {
        cout << "see" << endl;
    }
};

多继承

c++可以实现多重继承,但尽量不要采用,会增加设计和维护的复杂度,可以采用类似于java中的interface接口模式,就是用纯虚函数来实现多继承,会简化设计。
多继承简单语法:

class IosDeveloper {
public:
    void developIosApp() {
        cout << "I can develop ios" << endl;
    }
};

class AndroidDeveloper {
public:
    void developAndroidApp() {
        cout << "I can develop Android" << endl;
    }
};

class CrossPlatformDeveloper : public IosDeveloper, public AndroidDeveloper {
public:
    void developerH5() {
        cout << "I can develop H5" << endl;
    }
};

CrossPlatformDeveloper cpd;
cpd.developIosApp();
cpd.developAndroidApp();
cpd.developerH5();

多继承-虚函数

如果子类继承多个均有虚函数的父类,则子类会生成多张虚表。


b.png
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1. 析构函数和虚析构函数 如果基类的析构函数是虚的,那么它的派生类的析构函数都是虚的 这将导致:当派生类析构的时...
    杰伦哎呦哎呦阅读 7,202评论 0 2
  • 继承和多态 1. 继承的优缺点 优点:(1)子类可以灵活地改变父类中的已有方法;(2)能够最大限度的实现代码重用。...
    MinoyJet阅读 3,880评论 0 0
  • 一、多态 1. 什么是多态 多态按字面的意思就是多种形态。当类之间存在层次结构,并且类之间是通过继承关联时,就会用...
    送分童子笑嘻嘻阅读 1,693评论 0 0
  • 二十.多态与虚函数 多态:多态按字面的意思就是多种形态。当类之间存在层次结构,并且类之间是通过继承关联时,就会用到...
    b83dcb2e8b71阅读 3,746评论 0 1
  • 虚函数 在类的定义中,前面有 virtural 关键字的成员函数被称为虚函数。 virtual 关键字只用在类定义...
    Mitchell阅读 3,688评论 0 2

友情链接更多精彩内容