类和对象—多态(一)

简介: 类和对象—多态(一)

多态的概念

多态分为两类
静态多态:函数重载和运算符重载属于静态多态,复用函数名
动态多态:派生类和虚函数实现运行时多态
静态多态和动态多态的区别
静态函数的函数地址早绑定——编译阶段确定函数地址
动态函数的函数地址晚绑定——运行阶段确认函数地址
下面通过案例进行讲解多态

地址早绑定

#include
using namespace std;
class animal
{
    
public:
    void speak()
    {
    
        cout << "动物在说话" << endl;
    }
};
class cat :public animal
{
    
public:
    void dospeak()
    {
    
        cout << "小猫在说话" << endl;
    }
};
//地址早绑定 在编译阶段确定函数地址
//如果想执行让猫说话,那么这个函数地址就不能提前绑定了,需要在运行阶段进行绑定,地址晚绑定
void speak(animal& a)
{
    
    a.speak();
}
int main()
{
    
    cat b;
    speak(b);
}

地址晚绑定

使用关键字virtual,使父类函数变为虚函数

#include
using namespace std;
class animal
{
    
public:
//加virtual关键字使父类函数变为虚函数
    virtual void speak()
    {
    
        cout << "动物在说话" << endl;
    }
};
class cat :public animal
{
    
public:
//重写
//重写:函数返回类型,函数名,参数列表要与父类完全一致

     void speak()
    {
    
        cout << "小猫在说话" << endl;
    }
};
void speak(animal& a)
{
    
    a.speak();
}
int main()
{
    
    cat b;
    speak(b);
}

总结:
动态多态满足条件

  • 1.有继承关系
  • 2.子类重写父类的虚函数
    动态多态的使用
  • 1.父类的指针或者引用,指向子类对象

重写:函数返回类型,函数名,参数列表要与父类完全一致

原理剖析

在这里插入图片描述
当子类出些重写时,在虚函数表内子类重写函数的地址会将父类的虚函数覆盖

class animal
{
    
public:
     void speak()
    {
    
        cout << "动物在说话" << endl;
    }
};

像这样的空类,计算其内存大小,大小为一个字节

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

产生虚函数时,他的大小将变为4个字节,原因是当使用关键字virtual时,会产生一个虚函数表,里面存放的的时函数的地址,地址的大小为4个字节。

多态的优点以及实例

  • 代码组织结构清晰
  • 可读性强
  • 利于前期和后期的扩展以及维护
    实例:
#include
using namespace std;
class Calculator
{
    
public:
    int m_num1;
    int m_num2;
   virtual    int getResult()
   {
    
       return 0;
   }
};
class Add :public Calculator
{
    
public:
    int getResult()
    {
    
        return m_num1 + m_num2;
    }
};
class Sub :public Calculator
{
    
public:
    int getResult()
    {
    
        return m_num1 - m_num2;
    }
};
class Mul :public Calculator
{
    
public:
    int getResult()
    {
    
        return m_num1 * m_num2;
    }
};
int main()
{
    
    Calculator* abc;
    //加法
    abc = new Add;
    abc->m_num1 = 100;
    abc->m_num2 = 100;
    cout << abc->m_num1 << " + " << abc->m_num2 << " = " << abc->getResult() << endl;
    delete abc;
   //减法
    abc = new Sub;
    abc->m_num1 = 100;
    abc->m_num2 = 100;
    cout << abc->m_num1 << " - " << abc->m_num2 << " = " << abc->getResult() << endl;
    delete abc;
    //乘法
    abc = new Mul;
    abc->m_num1 = 100;
    abc->m_num2 = 100;
    cout << abc->m_num1 << " * " << abc->m_num2 << " = " << abc->getResult() << endl;
    delete abc;
    return 0;
}
相关文章
|
SQL 数据库
使用PD(PowerDesigner)图如何快速生成创建数据库表的SQL脚本并生成表在数据库
使用PD(PowerDesigner)图如何快速生成创建数据库表的SQL脚本并生成表在数据库
900 0
使用PD(PowerDesigner)图如何快速生成创建数据库表的SQL脚本并生成表在数据库
|
5月前
|
人工智能 PyTorch TensorFlow
AI界的"翻译官":ONNX如何让各框架模型和谐共处
还在为不同框架间的模型转换头疼?ONNX让你在PyTorch训练的模型可以无缝在TensorFlow部署,甚至能让模型在手机上飞速运行。本文带你了解这个AI领域的'瑞士军刀',轻松实现跨平台高性能模型部署。
262 12
|
算法 调度 UED
深入理解操作系统中的进程调度
【8月更文挑战第29天】操作系统是计算机系统的核心,负责管理硬件资源和提供各种服务。其中,进程调度是操作系统中至关重要的一环,它决定了哪些进程将获得处理器时间以及何时获得。本文将从进程调度的基本概念出发,探讨不同类型的调度算法及其优缺点,并通过实际代码示例加深理解。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供有价值的见解和知识。
211 8
|
11月前
|
Kubernetes Java 调度
Kubernetes中的Pod垃圾回收策略是什么
Kubernetes中的Pod垃圾回收策略是什么
|
SQL 缓存 Oracle
关系型数据库Oracle性能问题
【7月更文挑战第16天】
199 2
|
Web App开发 前端开发 JavaScript
灵魂拷问-前端的作用--chrome插件篇
本文会从浏览器插件应用场景切入,穿插插件基础能力和常见入口的介绍,核心回答如下三个问题:插件可以被使用在哪些场景?不同的使用场景我们的主要代码实现思路是怎样的?我们可以从哪些角度入手自己开发一款可以落地实用的浏览器插件?
|
资源调度 前端开发
文本,vitepress的使用,如何使用vitevitepress没有config.js该怎么办?这里使用vitepress进行手动配置,参考只爭朝夕不負韶華的文章
文本,vitepress的使用,如何使用vitevitepress没有config.js该怎么办?这里使用vitepress进行手动配置,参考只爭朝夕不負韶華的文章
|
存储 机器学习/深度学习 SQL
SQLSERVER存储过程语法详解
SQLSERVER存储过程语法详解
732 0
|
监控 网络协议 数据安全/隐私保护
手机socket套接字75秒超时问题及解决方案
手机socket套接字75秒超时问题及解决方案
336 0
|
缓存 网络协议 算法
《跟闪电侠学Netty》阅读笔记 - Netty入门程序解析(二)
《跟闪电侠学Netty》阅读笔记 - Netty入门程序解析
263 1