Chapter 30 多态

news2025/1/13 13:34:13

欢迎大家订阅【Python从入门到精通】专栏,一起探索Python的无限可能!

文章目录

  • 前言
  • 一、基本概念
  • 二、抽象类


前言

多态(Polymorphism)是面向对象编程中的核心概念,本章将详细讲解 Python 中多态的实现方式以及如何应用多态提高代码的可维护性。


本篇文章参考:黑马程序员

一、基本概念

定义:
多态是指同一个行为通过不同的对象能获得不同的结果或状态。

类型:
①鸭子类型(Duck Typing)
在 Python 中,我们不关注对象的具体类型,只关注对象是否具有某个方法或属性。如果一个对象能“像鸭子一样叫,像鸭子一样走”,那么即使它不是一个鸭子,Python 也能通过该对象执行操作。

class Bird:  
    def fly(self):  
        return "I can fly!"  

class Airplane:  
    def fly(self):  
        return "I am an airplane!"  

class Fish:  
    def swim(self):  
        return "I can swim!"  

def lets_fly(thing):  
    print(thing.fly())  

# 使用鸭子类型  
bird = Bird()  
airplane = Airplane()  

lets_fly(bird)      # 输出: I can fly!  
lets_fly(airplane)  # 输出: I am an airplane!  
# lets_fly(Fish())  # 会出错,因为 Fish 类没有 fly 方法

输出结果:
I can fly!
I am an airplane!

【分析】
lets_fly() 函数期望传入一个具有 fly() 方法的对象。Bird 和 Airplane 类都实现了 fly() 方法,因此它们可以作为参数传递给 lets_fly()。由于Fish 没有 fly() 方法,如果 Fish 被传入,程序运行后会报错。

②方法重写(Method Overriding)
子类可重写父类的方法,以实现不同的功能。当通过父类调用这个方法时,会调用子类的实现。

思考:那我们前两章讲到的继承的复写和多态的方法重写有什么区别呢?
答:在 Python 中,继承中的复写(通常指方法重写)和多态中的方法重写实际上是同一个概念,都是指在子类中重写父类的方法,但其上下文和侧重点有所不同。
在继承关系中,子类重写父类的方法,目的是给该方法提供不同的实现,以满足子类的特定需求。这是一种语言级别的概念,侧重于代码的结构和重用。
而多态中的“方法重写”更广泛地应用于设计模式,它关注的是运行时的行为选择,强调的是基于接口而不是固定实现,允许根据实际使用的对象类型动态地调用相应的方法。

class Animal:
    def speak(self):
        pass

class Dog(Animal):
    def speak(self):
        print("汪汪汪")

class Cat(Animal):
    def speak(self):
        print("喵喵喵")

def make_noise(animal: Animal):
    # 制造噪音,需要传入Animal对象
    animal.speak()

# 使用两个子类对象调用函数
dog = Dog()
cat = Cat()

make_noise(dog)
make_noise(cat)

输出结果:
汪汪汪
喵喵喵

【分析】
父类 Animal 定义了一个方法 speak(),而子类 Dog 和 Cat 重写了该方法,提供各自特定的实现。函数 make_noise() 接受一个 Animal 类型的参数,它可以接受任何 Animal 的子类实例(如 Dog 或 Cat)。这是因为子类是父类的特例,Dog 和 Cat 类是对 Animal 的扩展,因此任何 Dog 或 Cat 的实例都可以被视为 Animal 的实例。
在这里插入图片描述
多态常用于继承关系中,允许以父类进行定义和声明,而以子类实现具体功能,从而实现相同的行为但具有不同的状态。
例如,当一个函数的参数声明为父类对象时,实际上可以传入其子类的对象来执行相应的操作,这使得我们能够通过统一的接口来处理不同类型的对象,从而实现灵活的代码设计。

异同点:

特点鸭子类型方法重写
关注点对象的行为,而非其类型类的继承关系和方法的重写
灵活性高(只要对象实现了所需的方法)相对较低(依赖于明确的继承关系)
使用场景常用于接口实现或多态操作用于改变或扩展父类的行为

二、抽象类

抽象类:包含抽象方法的类。
抽象方法:没有具体实现的方法

这种设计的意义在于父类定义了应包含的方法,而具体的实现则由子类负责决定。

例如,空调可以制冷、制热和左右扫风。我们将这三个功能制定为空调制造的标准,不同的厂家必须根据这个标准各自实现功能。抽象类可以看作是这种标准,它包含了一些抽象方法,要求子类必须实现这些方法。
在这里插入图片描述
这种设计的作用主要体现在以下几个方面:

  • 顶层设计:抽象类提供了一个设计标准,以便子类进行具体实
  • 软性约束:抽象类对子类施加了一种约束,要求子类必须重写(实现)父类的一些方法
  • 配合多态:通过多态的机制,我们可以获得不同的工作状态
# 抽象类
class AC:
    # 制冷
    def cool_wind(self):
        pass

    # 制热
    def hot_wind(self):
        pass

    # 左右摆风
    def swing_l_r(self):
        pass


class Midea_AC(AC):
    def cool_wind(self):
        print("美的空调制冷")

    def hot_wind(self):
        print("美的空调制热")

    def swing_l_r(self):
        print("美的空调左右摆风")

class GREE_AC(AC):
    def cool_wind(self):
        print("格力空调制冷")

    def hot_wind(self):
        print("格力空调制热")

    def swing_l_r(self):
        print("格力空调左右摆风")

def make_cool(ac: AC):
    ac.cool_wind()

midea_ac = Midea_AC()
gree_ac = GREE_AC()

make_cool(midea_ac)
make_cool(gree_ac)

输出结果:
美的空调制冷
格力空调制冷

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1994001.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SQL Zoo 8.Using Null

以下数据均来自SQL Zoo 1.List the teachers who have NULL for their department.(列出所属部门为NULL的教师) select name from teacher where dept is null 2.Note the INNER JOIN misses the teachers with no department and the departments wit…

【Git】Git安装_配置

一、Git的安装 1. 下载Git 官方下载:访问Git的官方网站https://git-scm.com/,在首页找到下载链接。但需要注意的是,由于Git的服务器位于国外,直接下载可能会比较慢。镜像下载:推荐使用国内的镜像网站进行下载&#x…

【Canvas与艺术】蓝波纹白底黄星徽章

【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>蓝波纹白底黄星徽章</title><style type"text/css&qu…

《系统架构设计师教程(第2版)》第13章-层次式架构设计理论与实践-04-数据访问层设计

文章目录 1. 五种数据访问模式1.1 在线访问1.2 DAO1.3 DTO1.4 离线数据模式1.5 对象/关系映射 (O/R Mapping) 2. 工厂方法模式在数据访问层应用3 ORM、Hibernate与CMP2.0设计思想3.1 ORM3.2 Hibernate1&#xff09;概述2&#xff09; Hibernate的架构&#xff08;2023年的考题&…

SpringBoot统一功能处理——统一数据返回格式

目录 一、简单使用 二、存在的问题描述 三、优点 一、简单使用 统一的数据返回格式使用 ControllerAdvice 和 ResponseBodyAdvice 的方式实现 ControllerAdvice 表示控制器通知类。 添加类 ResponseAdvice , 实现 ResponseBodyAdvice 接口&#xff0c;并在类上添加 …

【jQuery】入门学习篇

文章目录 一、前言&#x1f680;&#x1f680;&#x1f680;二、jQuery简介及使用详解&#xff1a;☀️☀️☀️2.1 jQuery简介2.2 引入jQuery① 第一种引入方式&#xff1a;直接路径引入② 第二种引入方式&#xff1a;使用第三方CDN 后序还在更新中~~~三、总结&#xff1a;&am…

C++_基本语法笔记_类和对象

对于有Java基础的思想不过多记录&#xff0c;仅看语法 创建和封装对象 类似Java中的set操作&#xff0c;可以写一个赋值操作用于给实例化对象赋属性值 三种权限&#xff1a;public &#xff0c; protect &#xff0c;private Class和Struct区别 Struct默认为public权限cla…

SpringBoot统一功能处理——拦截器

目录 一、什么是拦截器&#xff1f; 二、拦截器使用 2.1 定义拦截器 2.2 注册配置拦截器 三、拦截器详解 3.1 拦截器的拦截路径配置 3.2 拦截器执行流程 一、什么是拦截器&#xff1f; 拦截器是Spring框架提供的核心功能之一, 主要用来拦截用户的请求, 在指定方法前后,…

C语言 | Leetcode C语言题解之第332题重新安排行程

题目&#xff1a; 题解&#xff1a; char* id2str[26 * 26 * 26];int str2id(char* a) {int ret 0;for (int i 0; i < 3; i) {ret ret * 26 a[i] - A;}return ret; }int cmp(const void* _a, const void* _b) {int **a (int**)_a, **b (int**)_b;return (*b)[0] - (*…

python网络爬虫使用代理

Python网络爬虫使用代理的实用指南 在网络爬虫的开发过程中&#xff0c;使用代理是一个非常重要的环节。代理不仅可以帮助爬虫绕过反爬虫机制&#xff0c;还能保护开发者的隐私。本文将介绍如何在Python中使用代理进行网络爬虫&#xff0c;包括基本的设置和示例代码。 1. 代理…

手机短视频素材网站有哪些啊?手机视频素材库网站分享

在当今的数字时代&#xff0c;智能手机与社交媒体平台的融合推动了手机短视频的兴起。这种形式的媒体已经渗透到我们的日常生活中&#xff0c;无论是作为娱乐手段、教育工具还是商业推广手段&#xff0c;优质的短视频都具有极高的吸引力和广泛的应用价值。因此&#xff0c;选择…

『 C++ 』智能指针 ( 万字梳理 )

文章目录 智能指针概念内存泄漏的危害RAII与智能指针智能指针的赋值auto_ptr 管理权转移auto_ptr 的对象悬空问题 unique_ptr 防拷贝unique_ptr 简单实现 shared_ptr 引用计数shared_ptr 简单实现shared_ptr 的循环引用问题与 weak_ptr 智能指针的自定义删除器 智能指针概念 智…

【独家原创】基于NRBO-Transformer多特征分类预测【24年新算法】 (多输入单输出)Matlab代码

【独家原创】NRBO-Transformer分类 Matlab代码 基于牛顿拉夫逊优化算法优化Transformer的数据分类预测&#xff0c;Matlab代码&#xff0c;可直接运行&#xff0c;适合小白新手 NRBO优化的超参数为&#xff1a;自注意力机制中的头数、正则化系数、初始化学习率 1.程序已经调试…

《知识点扫盲 · Redis 分布式锁》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…

数据库索引设计原则

1. 概述 索引是优化数据库性能最重要的工具之一。但是&#xff0c;创建过多的索引或索引错误的列也会对性能产生负面影响。因此&#xff0c;在设计索引时遵循一定的原则很重要。 2. 原则A - 根据工作负载创建索引 创建高效索引最重要的原则是根据您的工作负载而不是表结构创…

数据库连接池的深入学习

为什么需要数据库连接池&#xff1f; 正常操作数据库需要对其进行连接&#xff0c;访问数据库&#xff0c;执行sql语句&#xff0c;断开连接。 创建数据库连接是一个昂贵的过程&#xff0c;在高并发的情况下&#xff0c;频繁的创建数据库的连接可能会导致数据库宕机。 有了连…

Leetcode JAVA刷刷站(8)字符串转换整数

一、题目概述 二、思路方向 要实现这个功能&#xff0c;我们可以遵循以下步骤来编写 myAtoi 函数&#xff1a; 去除前导空格&#xff1a;使用循环或字符串的 trim() 方法&#xff08;虽然直接操作字符串更高效的方式是使用循环&#xff09;。检查符号&#xff1a;记录第一个非…

TGANet部分复现

Kvasir-SEG复现结果 M e t h o d m I o U m D S C R e c a l l P r e c i s i o n F 2 P r a N e t − − − − − − 0.9663704860609511 − − T G A N e t 0.8331 0.8982 0.9132 − − 0.9029 \begin{array}{lccccr} Method&mIoU&mDSC&Recall&Precision&a…

5、Linux : 网络相关

OSI七层网络模型 TCP/IP四层 概念模型 对应网络协议 应用层&#xff08;Application&#xff09; HTTP、TFTP, FTP, NFS, WAIS、 表示层&#xff08;Presentation&#xff09; 应用层 Telnet, Rlogin, SNMP, Gopher 会话层&#xff08;Session&#xff09; SMTP…

ICETEK-DM6437-AICOM——CPU定时器及直流电机控制中断控制

一、设计目的&#xff1a; 1.1 CPU定时器程序设计&#xff1b; 1.2 2直流电机程序设计&#xff1b; 1.3 外中断。 二、设计原理&#xff1a; 2.1 定时器的控制&#xff1a; 在DM6437&#xff08;是一种数字信号处理器&#xff0c;DSP&#xff09;上使用其内部定时器和中断来…