Python学习笔记12:进阶篇(二),类的继承与组合

类的继承

我们在编写一系列的类的时候,会发现这些类很相似,但是又有各自的特点和行为。在编写这些类的时候,我们可以把相同的部分抽象成一个基类,然后根据其他不同的特点和行为,抽象出子类,继承这个基类。通过继承,我们就可以方便的完整这一些列类的抽象。

示例

class Person:def __init__(self, name, age):self.name = nameself.age = ageself.country = "中国"  # 设置默认值def speak(self):print(f"大家好, 我的名字是{self.name}, 今年{self.age}岁,我来自{self.country}. 我讲的是普通话.")class GuangdongPerson(Person):def __init__(self, name, age):super().__init__(name, age)self.province = "广东"def speak(self):print(f"大家好, 我是{self.name}, 今年{self.age}岁,我来自{self.country}{self.province}. 我会讲广东话.")class TaiwanPerson(Person):def __init__(self, name, age):super().__init__(name, age)self.province = "台湾"def speak(self):print(f"大家好, 我是{self.name}, 今年{self.age}岁,我来自{self.country}{self.province}. 我会讲台湾话.")zhang_san = GuangdongPerson("张三", 18)
li_si = TaiwanPerson("李四", 20)
zhang_san.speak()
li_si.speak()

在这里插入图片描述

例子不能说讲的很好哈,能理解就行。像我之前推荐的书,书里的举例是car,其他教程里面可能还有shape。但是举什么例子不重要,重要的是了解什么是继承,继承能干什么,有什么意义,好处和坏处是什么。

什么是继承

继承是一种面向对象编程的基本特性,它允许我们定义一个类(称为子类或派生类),该类可以从另一个已经存在的类(称为父类或基类)那里继承属性和方法。这意味着子类会自动获得父类的所有功能,同时还可以定义或修改自己的属性和方法,或者覆盖父类的方法来实现特定的功能。
在示例中,我们先定义了一个父类Person。而后定义了GuangdongPerson和TaiwanPerson两个类,这两个类定义了自己的属性,继承Person类的三个属性和函数,最后我们重写来自于父类的方法。

为什么要继承

继承的主要目的是促进代码的复用性、模块化和易于维护。通过继承,我们可以创建一个通用的基类来封装共同的行为,然后为具有特殊需求的子类添加或修改功能,而不需要重复编写相同的代码。

继承有哪些缺点

没有完美无缺的事物,继承在提供便利的同时也带来一些问题:

  1. 过度耦合: 如果子类过于依赖父类的实现细节,当父类发生变化时,可能会对子类产生意想不到的影响。这种紧密的耦合关系使得代码难以修改和维护。
  2. 复杂度增加: 随着继承层次的加深,类结构会变得更加复杂。开发者需要跟踪多个层级的继承关系,理解每个类的功能和责任变得困难。
  3. 灵活性降低: 继承是静态的,一旦一个类继承了另一个类,这种关系就固定下来,不能在运行时改变。这限制了代码的灵活性和可重用性。
  4. 菱形问题(Diamond Problem): 在多继承的语言中,如果多个父类拥有同一个方法或属性,子类可能会遇到冲突,不确定应该使用哪一个父类的实现。
  5. 代码可读性和可维护性下降: 过度使用继承可能导致“类爆炸”,即类的数量急剧增加,这会使代码库变得庞大且难以管理。

什么时候用继承

选择使用继承主要取决于以下几个场景和原则:

  1. 共享行为和属性: 当多个类具有相似的属性和行为时,可以考虑使用继承。一个基类可以定义这些共性部分,而子类则专注于差异化的部分。例如,各种动物类可能都共享“移动”和“吃”的行为,但具体实现各不相同。
  2. 代码复用: 继承允许子类重用父类的代码,减少重复代码,提高代码的可维护性。如果多个类需要执行相同的操作或拥有相同的属性,通过继承可以避免重复定义这些内容。
  3. 层次结构: 当类之间存在明显的层次或分类关系时,继承是自然的选择。例如,在一个图形界面库中,Widget类可以作为基类,而Button、TextBox等具体控件类继承自它,形成一个清晰的层次结构。
  4. 接口一致性: 如果希望一组类提供一致的接口给外部使用者,即使它们的内部实现不同,也可以通过继承来实现。这使得客户端代码可以以统一的方式处理这些类的实例。
  5. 抽象类和接口: 使用抽象基类(ABC)可以定义一个接口或模板,强制子类必须实现特定的方法,从而保证了设计的一致性和约束性。

通常继承和组合两个方式来实现某些属性和行为的复用。当两个类之间的关系不是“is-a”的关系,而是“has-a”的关系时,组合(包含另一个类的实例)通常是更好的选择。

组合

当我们需要复用某些类的属性和行为的时候,使用继承的方式和客观事实相违背,但是有另一个符合客观事实的复用方式,就是组合。简单理解为定义一个类A的时候,将另一个已存在的类BA自己本身的一些属性和行为放到一起组合成类A

什么是组合

组合(Composition)是面向对象编程中的另一种重要的设计原则,用于表示“整体-部分”(has-a)的关系,而不是继承所表达的“是一个”(is-a)的关系。在组合中,一个类的对象可以包含另一个或多个类的对象作为其属性,以此来实现更复杂的功能或结构,而无需通过继承来扩展类的功能。

组合的好处

  1. 灵活性:相比于继承,组合提供了更高的灵活性。因为对象可以在运行时动态地添加、替换或移除其组成部分,这使得系统更加灵活,更容易适应变化。
  2. 明确的责任分配:每个对象负责自身的功能,整体对象通过协调其组成部分来完成更复杂的任务,这有助于遵循单一职责原则。
  3. 避免继承的局限性:组合可以有效避免继承的缺点,如多重继承的菱形问题、过深的继承层次导致的复杂性等。
  4. 促进代码重用:通过将复杂对象构建为简单对象的组合,可以在不同的上下文中重用这些简单对象,而不需要创建新的继承层次。

什么时候用组合

  1. 当一个类需要使用另一个类的功能,但两者之间不存在“是一种”关系时。
  2. 当需要在运行时动态地调整或扩展对象的功能时。
  3. 当想要设计出高度可配置和可变的对象结构时。

示例

class Engine:def start(self):print("Engine started.")class ElectricEngine(Engine):def start(self):print("Electric Engine started.")class InternalCombustionEngine(Engine):def start(self):print("Internal Combustion Engine started.")class Car:def __init__(self):self.engine = Engine()  # Car类包含了一个Engine对象,这就是组合def set_engine(self, engine):self.engine = enginedef start_car(self):self.engine.start()  # 通过组合的Engine对象来启动汽车my_car = Car()
# 普通引擎
my_car.start_car()
my_car.set_engine(ElectricEngine())
# 电动引擎
my_car.start_car()
my_car.set_engine(InternalCombustionEngine())
# 汽油引擎
my_car.start_car()

在这里插入图片描述
在例子中,我们需要定义一个汽车类。汽车可以通过引擎启动。所以我们先定义一个引擎类来启动汽车。这个时候问题来了,汽车继承引擎,是不是很奇怪。汽车和引擎很明显不是is-a的关系,而是has-a的关系,所以我们需要使用组合的方式,将引擎类作为汽车的一个属性封装到汽车类中。
最后,使用不同的引擎,启动方法的实现也不一样,成功完成目标需求。

作业:

  1. 了解is-a和has-a,举出例子,分别使用继承和组合的方式实现类的复用。

下课!!!

点赞收藏关注,感谢支持!!!

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

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

相关文章

上海AI Lab推出8B模型,奥数成绩媲美GPT-4

只用1/200的参数,就能让大模型拥有和GPT-4一样的数学能力? 复旦大学和上海AI实验室的研究团队刚刚研发出了一款具有超强数学能力的模型。 这款模型名为MCTSr,以Llama 3为基础,参数量只有8B,却在奥赛级别的题目上取得了…

Nuxt快速学习开发 -- Nuxt3配置

Nuxt配置 nuxt.config.ts文件位于 Nuxt 项目的根目录下,可以覆盖或扩展应用程序的行为 使用可组合项,这些变量会暴露给应用程序 //nuxt.config.ts import { fileURLToPath } from "url"; ​ export default defineNuxtConfig({alias: {//配置…

Java线程池基本概念

全局和局部线程池 全局线程池 在Spring框架中,全局线程池如ThreadPoolTaskExecutor通常是作为Spring Bean存在的,它们的生命周期由Spring容器管理。当Spring容器关闭时,这些线程池也会被适当地清理和关闭。因此,开发者通常不需要手…

DDD架构和微服务初步实现

本次记录的是微服务的初步认识和DDD架构的初步实现和思路,在之前的发布里,对Javaweb进行了一次小总结,还有一些东西,不去详细理解说明了,下面开始我对微服务的理解。 什么是微服务? 在刚刚开始学习的时候…

Study--Oracle-03-数据库常规操作

一路走来,所有遇到的人,帮助过我的、伤害过我的都是朋友,没有一个是敌人。 一、oracle 版本及主要功能 二、数据安装完成后常用操作SQL 1、检查数据库监听状态 监听的常用命令 启动:[oracleoracle u01]$ lsnrctl stop 停止&am…

2024信息系统、信号处理与通信技术国际会议(ICISPCT2024)

2024信息系统、信号处理与通信技术国际会议(ICISPCT2024) 会议简介 2024国际信息系统、信号处理与通信技术大会(ICISPCT2024)将在青岛隆重开幕。本次会议旨在汇聚全球信息系统、信号处理和通信技术领域的专家学者,共同探索行业…

【记录46】【案例】echarts 柱状图

echarts环境4.1.0 <template><div id"threefour"></div> </template> <script> import * as echarts from "echarts" export default {name:"",components:{},data(){return {}},methods:{getdata(){var myChart…

《平衡小车控制系统》电子设计大赛校赛感悟

我们学校举行了一次电子设计大赛选拔赛&#xff0c;虽然我们在测试的时候全部都可以完成&#xff0c;最后考核的时候因为方案选择问题以及各种设计逻辑等原因没能成功晋级&#xff0c;但我能从这次备赛中学到很多东西&#xff0c;遂分享一下&#xff0c;与广大网友交流经验。&a…

英伟达发布开源模型Nemotron-4 340B

&#x1f680; 英伟达发布开源模型Nemotron-4 340B 摘要&#xff1a;英伟达最新发布的开源模型Nemotron-4 340B&#xff0c;可能彻底改变大语言模型&#xff08;LLM&#xff09;训练方式。该模型支持多种自然语言和编程语言&#xff0c;使用9万亿个token训练&#xff0c;高达9…

Day 26:2288. 价格减免

Leetcode 2288. 价格减免 句子 是由若干个单词组成的字符串&#xff0c;单词之间用单个空格分隔&#xff0c;其中每个单词可以包含数字、小写字母、和美元符号 ‘$’ 。如果单词的形式为美元符号后跟着一个非负实数&#xff0c;那么这个单词就表示一个 价格 。 例如 “$100”、…

Windows系统部署本地SQL_Server指引

Windows系统部署本地SQL_Server指引 此指引文档环境为Windows10系统&#xff0c;部署SQL_Server 2019为例&#xff0c;同系列系统软件安装步骤类似。 一、部署前准备&#xff1b; 下载好相关镜像文件&#xff1b;设备系统启动后&#xff0c;将不必要的软件停用&#xff0c;避…

【Linux】shell——条件判断test,各种运算符,expr

条件判断——test 真——0 假——1 test expression or [ expression ] 整数运算符 字符串运算符 -z 长度是否为0 -n 长度是否不为0 str1 str2 str1 ! str2 补 &&-->逻辑与&#xff0c;前面为真后面才会执行 || -->逻辑或&#xff0c;前面为假后面才…

VirtFuzz:一款基于VirtIO的Linux内核模糊测试工具

关于VirtFuzz VirtFuzz是一款功能强大的Linux内核模糊测试工具&#xff0c;该工具使用LibAFL构建&#xff0c;可以利用VirtIO向目标设备的内核子系统提供输入测试用例&#xff0c;广大研究人员可以使用该工具测试Linux内核的安全性。 工具要求 1、Rust&#xff1b; 2、修补的Q…

线代的学习(矩阵)

1.矩阵的乘法 矩阵实现满足&#xff1a;内标相等 矩阵相乘之后的结果&#xff1a;前行后列 需要注意&#xff1a;1.矩阵的乘法不具有交换律&#xff1a;AB!BA 2.矩阵的乘法满足分配律&#xff1a;A(BC) AB AC 抽象逆矩阵求逆矩阵 方法1.凑定义法、 方法2.长除法 数字型矩阵…

算法金 | 一个强大的算法模型:t-SNE !!

大侠幸会&#xff0c;在下全网同名「算法金」 0 基础转 AI 上岸&#xff0c;多个算法赛 Top 「日更万日&#xff0c;让更多人享受智能乐趣」 t-SNE&#xff08;t-Distributed Stochastic Neighbor Embedding&#xff09;是一种用于降维和数据可视化的非线性算法。它被广泛应用于…

LeetCode 算法:合并两个有序链表 c++

原题链接&#x1f517;&#xff1a;合并两个有序链表 难度&#xff1a;简单⭐️ 题目 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例 1&#xff1a; 输入&#xff1a;l1 [1,2,4], l2 [1,3,4] 输出&#xff1a;…

AI智能盒子助力中钢天源设备工厂升级安全防护

中钢集团安徽天源科技股份有限公司成立于2002年3月27日,是中央企业中国中钢股份有限公司控股的上市公司&#xff0c;主导产品为永磁铁氧体器件、钕铁硼器件、四氧化三锰、锶铁氧体预烧料及各类磁选机等。 在中钢天源智能化升级过程中&#xff0c;采用并定制开发一系列厂区安全…

QT day02

思维导图 UI界面设计 设置登录界面&#xff0c;输入账号、密码&#xff0c;登录/取消 按钮 使用手动连接&#xff0c;将登录框中的取消按钮使用第二中连接方式&#xff0c;右击转到槽&#xff0c;在该槽函数中&#xff0c;调用关闭函数 将登录按钮使用qt4版本的连接到自定义…

Python期末复习题库(上)

1. (单选题) Python源程序的扩展名为&#xff08; A &#xff09; A. py B. c C. class D. ph 2. (单选题) 下列&#xff08; A &#xff09;符合可用于注释Python代码。 A. # B. */ C. // D. $ 3. (单选题)下列关于Python 语言的特点的说法中&#xff0c;错误的是&#xf…

【Linux基础IO】常见的对文件操作的函数、文件描述符fd、访问文件的本质分析

目录 fopen函数 chdir函数 fclose函数 fwrite和fread函数 open函数 umask函数 write函数 read函数 close函数 文件描述符fd 进程访问文件的本质分析 fopen函数 参数mode&#xff1a; w方式打开文件&#xff1a;1、如果被打开文件不存在&#xff0c;系统会在使用fopen函…