多态

多态:静态多态,动态多态

静态多态:函数重载、运算符重载

动态多态:派生类和虚函数实现运行时多态

静态多态与动态多态的区别:

静态多态的函数地址早绑定-编译阶段确定函数地址

动态多态的函数地址晚绑定-运行阶段确定函数地址

多态满足条件:
有继承关系

子类重写父类的虚函数

多态使用条件:
父类指针或引用指向子类对象

 

#include <iostream>


class Animal{
    public:
    void speak(){
        std::cout<<"动物在说话"<<std::endl;
    }
    virtual void speak1(){
        std::cout<<"动物1在说话"<<std::endl;
    }
};

class Dog:public Animal{
    public:
    void speak(){
        std::cout<<"小狗在说话"<<std::endl;
    };
    void speak1(){
        std::cout<<"小狗1在说话"<<std::endl;
    };
};

class Cat:public Animal{
    public:
    void speak(){
        std::cout<<"小猫在说话"<<std::endl;
    };
     void speak1(){
        std::cout<<"小猫1在说话"<<std::endl;
    };
};

void doSpeak(Animal &animal){
    animal.speak();
    animal.speak1();

}

int main(){
    Dog dog;
    doSpeak(dog);
    Cat cat;
    doSpeak(cat);
    return 0;
}

  

多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用到子类的析构代码

解决方法:将父类中的析构函数改为虚析构或者纯虚析构

虚析构和纯虚析构共性:

   可以解决父类指针释放子类对象

   都需要有具体的函数实现

虚析构和纯虚析构的区别:
   如果是纯虚析构 ,该类属于抽象类,无法实例化对象

虚析构语法:

virtual ~类名(){}

 

纯虚析构语法:

virtual ~类名()=0;
类名::~类名(){}

利用虚析构函数可以解决 父类指针释放子类对象时不干净的问题

#include <iostream>

class Animal{
public:
     Animal(){
        std::cout<<"Animal 构造函数"<<std::endl;
     }
     //虚析构函数
     virtual ~Animal(){
        std::cout<<"Animal 析构函数"<<std::endl;
     }
    //纯虚函数
     virtual void speak()=0;
};

class Cat:public Animal{
    public:
    Cat(std::string name){
        m_Name= new std::string(name);
    }
    void speak(){
        std::cout<<"小猫在说话"<<std::endl;
    }
    ~Cat(){
        if (m_Name!=nullptr){
            std::cout<<"Cat 析构函数"<<std::endl;
            delete m_Name;
            m_Name=nullptr;
        }
    }
    std::string *m_Name=nullptr;
};



void doSpeak(Animal &animal){
   animal.speak();
}

int main(){

    //父类指针指向子类对象
    Animal *anmial=new Cat("Tom");
    // 父类指针在析构时候 不会调用子类中析构函数,导致子类如果有堆区数据,释放不干净
    anmial->speak();
    delete anmial;
    anmial=nullptr;
}

  

 

posted @ 2024-03-31 18:51  FigureOut  阅读(1)  评论(0编辑  收藏  举报