首页 > 编程语言 >【C++】多态

【C++】多态

时间:2024-08-01 16:54:32浏览次数:24  
标签:虚表 函数 派生类 多态 C++ 基类 重写

什么是多态?

就是不同对象在完成某个行为时,会产生不同的形态

多态分为静态多态和动态多态

静态多态

在编译期间就确定执行哪个函数,主要包含:函数重载和运算符重载

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

动态多态

派生类和虚函数实现的运行阶段的多态

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

函数重写(覆盖):

子类重新定义父类中有相同名称,返回值和参数的虚函数,只在继承关系中出现

基本条件:基类中被重写的函数必须是虚函数,并且派生类和基类中重写的函数和被重写的函数返回值 函数名 函数参数必须一致

函数隐藏

在派生类中,只要和基类函数名字相同,不是重写就是隐藏

#include<iostream>
using namespace std;

class Father{
public:
    virtual void fun(){
        cout<<"Father"<<endl;
    }
};

class Son : public Father{
public:
    //派生类的 virtual可以省略,基类必须是虚函数
    virtual void fun(){
         cout<<"Son"<<endl;
    }
    //函数隐藏
    void fun(int a){}

};

添加

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

多态的满足条件:继承,子类重写父类的虚函数

多态的实现(原理)

为了实现 C++ 的多态,C++ 使用了一种动态绑定技术。这个技术的核心是虚函数表。

虚函数表

下面介绍虚函数表是如何实现动态绑定的:

  • 每个包含虚函数的类都包含一个虚表(存放粗函数指针的数组)。
  • 当一个类 B 继承 类 A 时,会继承类 A 的函数的调用权。
  • 所以说如果一个基类包含虚函数,那么其派生类也可调用这些虚函数,这些派生类也有属于自己的虚表。
  • class A {
    public:
        virtual void vfun1(){}
        virtual void vfun2(){}
        void fun1(){}
        void fun2(){}
    
    };
    //类 A 包含虚函数 vfun1 vfun2,故 A 有自己的虚函数表
    
    class B : public A{
    public:
    };
    //类 B 继承了类 A,故 B 也有属于自己的虚函数表

    下面展示一下类 A 的虚表指示图

  • 虚表是一个存放虚函数指针的数组,其内的元素是虚函数指针,每个元素对应每个虚函数的函数指针

  • 需要注意的是:普通函数(即非虚函数)的调用不会经过虚表,所以虚表中并不包含普通函数的函数指针

  • 虚表内的条目,即虚函数指针的赋值发生在编译器编译阶段,也就是说在代码的编译阶段,虚表就可以构造出来了

简述一下多态实现的整体过程

假设 存在基类全价买票类,有一个学生类半价买票,退伍军人免费
class Purchase{
public:
    virtual void fun(){ 
        cout<<"全价买票"<<endl;
    }
};

class Student : public Purchase{
public:
    virtual void fun(){
        cout<<"半价"<<endl;
    }
};

class ArmyPerson : public Purchase{
public:
    virtual void fun(){
        cout<<"免费"<<endl;
    }
};

void vfun(Purchase* p){
    p->fun();
}

int main(){
    Student s;
    vfun(&s);
    ArmyPerson a;
    vfun(&a);
}

学生类继承了购买类,也就有自己的虚表和虚表指针,重写了fun函数,学生类中的虚表指针就会指向学生类的 fun 函数,而不是购买类的。军人类和学生类,加入存在大众类继承购买类,不重写 fun 函数,那么执行结果就是全价购买。这样在调用 vfun(s) 函数时满足基类指针指向子类对象,调用子类虚表指针,到虚表中执行相应的函数。军人类同理,这样就实现了不同人群买票的不同价位,也就是多态的定义,不同对象执行某一任务会有不同的状态。

抽象类

包含纯虚函数的类就是抽象类,抽象类不能实例化对象

纯虚函数就是虚函数 = 0

总结

允许不同的对象通过同一个函数实现不同的行为,基类的代码可以被派生类复用,并且能根据不同的派生类实现不同的行为,这就是多态。

标签:虚表,函数,派生类,多态,C++,基类,重写
From: https://blog.csdn.net/2301_77562764/article/details/140849311

相关文章

  • 【C++】1.C++入门(1)
    文章目录1.C++关键字2.命名空间2.1命名空间定义2.2命名空间使用3.C++输入&输出4.缺省参数4.1缺省参数概念4.2缺省参数分类5.函数重载5.1函数重载概念5.2C++支持函数重载的原理--名字修饰(nameMangling)1.C++关键字C++总计63个关键字,C语言32个关键字2.......
  • 深入探讨C++编译期多态:提高效率与安全性
    目录一、引言二、编译期多态技术详解函数重载(FunctionOverloading)运算符重载(OperatorOverloading)模板元编程(TemplateMetaprogramming)使用std::enable_if实现条件模板特化使用ifconstexpr实现编译期条件分支SFINAE(SubstitutionFailureIsNotAnError)CRTP(Curious......
  • 基于Django的超市小程序+47822(免费领源码)可做计算机毕业设计JAVA、PHP、爬虫、APP、小
    基于django超市小程序摘 要随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱,超市小程序被用户普遍使用,为方便用户能够可以随时进行超市小程序的数据信息管理,特开发了基于djan......
  • 移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——1.c++入门(2)
    1.函数重载C++⽀持在同⼀作⽤域中出现同名函数,但是要求这些同名函数的形参不同,可以是参数个数不同或者类型不同。这样C++函数调⽤就表现出了多态⾏为,使⽤更灵活。C语⾔是不⽀持同⼀作⽤域中出现同名函数的。#include<iostream>usingnamespacestd;//1、参数类型不同......
  • C++11 中的 python 海象运算符 (:=) 等效吗?
    最近我在Python中经常使用:=运算符,这样:ifmy_object:=SomeClass.function_that_returns_object():#dosomethingwiththisobjectifitexistsprint(my_object.some_attribute)问题有没有办法在c++11中做到这一点而不使用stdlib?例如......
  • Dev-c++萌新学习福利1
    萌新福利重磅来袭作品成本6999元!!!清北互联团队编写课程,本人不收费。亏本买卖,良心服务,同嫂无欺。本作品喜欢当我粉丝吧!!!外加点赞!!!......
  • 【C++第十章】String
    【C++第十章】学习StringSTL介绍......
  • c++function bind包装器详解
    包装器function包装器function包装器也叫适配器,c++中的function本质是一个类模板,也是一个包装器,那么为什么需要包装器呢。template<classF,classT>TuseF(Ff,Tx){staticintcount=0;cout<<"count:"<<++count<<endl;cout<<"count:&q......
  • Java--多态和抽象类
    目录多态实现多态的条件重写重写和重载静态绑定和动态绑定向上转型和向下转型向上转型向下转型instanceof抽象类多态在Java中多态就是在完成一件事的时候,不同人去完成产生的结果不同比方说打印,我们就是要打印一个东西,如果我们交给彩色打印机,那么会打印出来彩色......
  • 三种语言实现计算二进制中1的个数(C++/Python/Java)
    题目给定一个长度为n的数列,请你求出数列中每个数的二进制表示中1的个数。输入格式第一行包含整数n。第二行包含n个整数,表示整个数列。输出格式共一行,包含n个整数,其中的第i个数表示数列中的第i个数的二进制表示中1的个数。数据范围1≤n≤100000,0≤数列中......