• C++——多态


    多态是有继承关系的类对象调用相同的函数,会有不同的结果。例如,普通人买高铁票不打折,学生打75折,儿童免费,这种情况就适合使用多态

    虚函数

    被virtual修饰的类成员函数

    1. class A{
    2. public:
    3. virtual void fun() {}
    4. };

    这里的virtual和类的虚继承没有任何关系,只是使用了相同的关键词

    多态的条件

    1.必须使用基类的指针或引用调用基类的虚函数

    2.派生类中必须对被调用的基类中的虚函数进行重写

    3.重写的条件:派生类中函数的返回值,函数名和参数列表与基类中的虚函数相同(三同)

    (派生类中重写的函数可以不加virtual,因为该成员函数从父类中继承了virtual属性,但是建议加上更规范)

    1. class Person {
    2. public:
    3. virtual void buy_ticket()
    4. {
    5. cout << "full price ticket" << endl;
    6. }
    7. };
    8. class Student :public Person {
    9. public:
    10. virtual void buy_ticket()
    11. {
    12. cout << "half price ticket" << endl;
    13. }
    14. };
    15. void test1()
    16. {
    17. Person Tom;
    18. Student Mike;
    19. Person p1 = Mike;
    20. p1.buy_ticket();
    21. Person* pp = &Mike;
    22. pp->buy_ticket();
    23. Person& p = Mike;
    24. p.buy_ticket();
    25. //full price ticket
    26. //half price ticket
    27. //half price ticket
    28. }

    final

    final修饰虚函数表示这个虚函数不能再被重写

    1. class Person {
    2. public:
    3. virtual void buy_ticket()
    4. {
    5. cout << "full price ticket" << endl;
    6. }
    7. };
    8. class Student :public Person {
    9. public:
    10. void buy_ticket() final
    11. {
    12. cout << "half price ticket" << endl;
    13. }
    14. };

    override

    override修饰派生类虚函数,可以检查派生类虚函数是否重写了积累的虚函数,如果没有就会报错

    1. class Person {
    2. public:
    3. virtual void buy_ticket()
    4. {
    5. cout << "full price ticket" << endl;
    6. }
    7. };
    8. class Student :public Person {
    9. public:
    10. virtual void buy_ticket()override
    11. {
    12. cout << "half price ticket" << endl;
    13. }
    14. };

    重写的特例

    协变(了解)

    重写的的函数返回值不同,且基类的函数返回值是基类对象的指针或引用,派生类的函数返回值是派生类对象的指针或引用。

    注意:只要基类和派生类的返回值是继承关系即可,所以函数返回值不一定只是函数所在类的对象的指针或引用

    1. class A {
    2. };
    3. class B :public A {
    4. };
    5. class Person {
    6. public:
    7. virtual const A& fun()
    8. {
    9. return A();
    10. }
    11. };
    12. class Student :public Person {
    13. public:
    14. virtual const B& fun()
    15. {
    16. return B();
    17. }
    18. };

    析构函数

    析构函数需要重写吗?看下面这个例子

    1. class Person {
    2. public:
    3. ~Person() { cout << "~Person()" << endl; }
    4. };
    5. class Student : public Person {
    6. public:
    7. ~Student() { cout << "~Student()" << endl; }
    8. };
    9. void test2()
    10. {
    11. Person* p = new Student;
    12. delete p;
    13. //~Person()
    14. }

    由于没有重写,释放空间p的时候,只调用了Person的析构函数。造成内存泄漏

    那么析构函数怎么重写?函数名不一样这怎么办?

    编译器对析构函数进行处理,析构函数名都是destructor,我们只要在析构函数前加virtual就可以

    1. class Person {
    2. public:
    3. virtual ~Person() { cout << "~Person()" << endl; }
    4. };
    5. class Student : public Person {
    6. public:
    7. virtual ~Student() { cout << "~Student()" << endl; }
    8. };
    9. void test3()
    10. {
    11. Person* p = new Student;
    12. delete p;
    13. //~Student()
    14. //~Person()
    15. }

  • 相关阅读:
    【计算机视觉】图像预处理
    element ui 里面只能输入input 数字 +限制长度
    如何实现暗示文本顺序
    项目运行状况不断,怎么办?看看企业级监控项目Skywalking吧
    wpf devexpress实现输入验证使用验证规则
    【医学影像数据处理】 Dicom 文件格式处理汇总
    Anaconda、conda、pip的区别
    HBase 记录
    【游戏引擎之路】登神长阶(七)——x86汇编学习:凡做难事,必有所得
    Aspose.Words for .NET查找和替换教程——使用元字符查找和替换文本
  • 原文地址:https://blog.csdn.net/weixin_74269833/article/details/133652034