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

news2024/11/14 13:55:39

类的继承

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

示例

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age
        self.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 = engine

    def 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.coloradmin.cn/o/1858331.html

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

相关文章

python实训day4

1、查看数据库的版本 2、查看当前用户 3、查看当前数据库 4、计算表达式的结果; 任何一个数据库,无论大小,都首先是一个超级计算器 5、查看当前MySQL环境中所有的数据库; 系统数据库(只能看)和自定义数据库(任何操作) 6、先建数据库 gaoming 7、如果表已经存在,则创建不能成功 …

刷题之小欧的平均数(卡码网)

小欧的平均数 这道题不看解析的话完全没有思路&#xff0c;连题目都没读明白&#xff0c;甚至看了评论答出来了还是不知道为什么&#xff0c;有知道的朋友可以教教我 #include<iostream> using namespace std;int main() {int x,y,z;cin>>x>>y>>z;//…

oracle12c到19c adg搭建(六)切换后12c备库服务器安装19c软件在19c主库升级数据字典后尝试同步

一、安装19c软件 参考文章oracle12c到19c adg搭建&#xff08;三&#xff09;oracle19c数据库软件安装 二、原主库尝试通过19c软件启动数据库 2.1复制12c的相关参数文件和密码文件到19c目录 注意:密码文件需要从已切换主库19c传过来 [oracleo12u19p ~]$ cd /u01/app/oracle…

亲测有效!性能压测异常竟能自动化分析!

性能压测是一种评估系统运行效率和稳定性的方法&#xff0c;通过模拟真实的使用场景和负载条件&#xff0c;对系统进行压力测试和负载测试&#xff0c;并对测试结果进行分析&#xff0c;以评估系统的性能&#xff0c;其中性能压测结果分析是性能压测的重要环节。以往的性能压测…

ONLYOFFICE 桌面编辑器 8.1 发布:全新功能齐备的 PDF 编辑器、丰富的幻灯片版式

前言 在当前数字化迅速发展的时代&#xff0c;办公软件套件作为企业日常工作的核心工具&#xff0c;其功能的全面性和使用的便捷性直接影响着工作效率和团队协作的效果。在众多办公软件中&#xff0c;ONLYOFFICE凭借其卓越的性能和丰富的功能&#xff0c;备受用户青睐。近日&a…

深入了解 GPT-4 和 ChatGPT 的 API---OpenAI Playground

文章目录 基本概念OpenAI API 提供的可用模型在 OpenAI Playground 中使用 GPT 模型 掌握GPT-4 和 ChatGPT 的 API 的使用方法&#xff0c;以便有效地将它们集成到 Python 应用程序中。首先&#xff0c;需要了解 OpenAI Playground。这将使你在编写代码之前更好地了解模型。接着…

React尚硅谷115-126(setState、Hooks、Fragment、context、组件优化、renderprops

122&#xff0c;context 只能用value传&#xff0c;可以传对象&#xff0c;字符串 一种组件间通信方式, 常用于【祖组件】与【后代组件】间通信 使用&#xff1a; 创建Context容器对象&#xff1a; const XxxContext React.createContext() 渲染子组时&#xff0c;外面包…

MIT6.s081 2021 Lab Utilities

Boot xv6 按照示例切换到 util 分支后&#xff0c;看到目录下包含 Makefile 文件&#xff0c;执行 make qemu 即可。 sleep 思路 借助系统调用 sleep 实现一个命令行程序&#xff0c;关键是要找到封装了系统调用的 C 函数的位置&#xff0c;根据提示&#xff1a; … user/u…

软件测试--花了我一周时间整理各种测试模板

2024软件测试面试刷题&#xff0c;这个小程序&#xff08;永久刷题&#xff09;&#xff0c;靠它快速找到工作了&#xff01;&#xff08;刷题APP的天花板&#xff09;-CSDN博客跳槽涨薪的朋友们有福了&#xff0c;今天给大家推荐一个软件测试面试的刷题小程序。https://blog.c…

北邮《计算机网络》MAC子层笔记

文章目录 缩写复习MAC层所在层次动态分配信道算法们的简要介绍信道的五条基本假设多路访问的协议&#xff08;理论上的协议&#xff09;aloha协议CSMA协议其他冲突避免协议无线局域网协议 &#xff0c;MACA 以太网协议802.3&#xff08;实际协议&#xff0c;刚刚是理论&#xf…

光猫BOB 功率调测误差分析与校验指南

DDM&#xff08;Digital Diagnostic Monitoring&#xff09;数字诊断监控技术&#xff0c;是光模块中使用的技术&#xff0c;以便用户能够监控光模块的实时参数。这些参数包括工作温度、工作电压、工作电流、发射和接收光功率等&#xff0c;还可以显示模块的常规波长、速率、传…

在Vue表单中设置缺省值

有个需求&#xff0c;在新增记录的时候&#xff0c;打开新增页面&#xff0c;员工姓名处获取到当前登录用户的用户名&#xff0c;并将其设置为缺省值。 /** 新增按钮操作 */handleAdd() {this.reset();this.open true;// this.form.employeeName this.$store.state.user.name…

计算机网络 DHCP以及防护

一、理论知识 1.DHCP&#xff1a;用于在网络中自动分配IP地址及其他网络参数&#xff08;如DNS、默认网关&#xff09;给客户端设备。 2.VLAN&#xff1a;逻辑上的局域网分段&#xff0c;用于隔离和管理不同的网络流量。 3.DHCP地址池&#xff1a;为每个VLAN配置不同的DHCP地…

【语义分割系列】基于cityscape的DDRNet算法

基于cityscape的DDRNet算法 前言 DDRNet是专门为实时语义分割设计的高效主干。该模型由两个深度分支组成,在这两个分支之间执行多次双边融合,并且还设计了一个新的上下文信息抽取器,名为深度聚合金字塔池模块(DAPPM),用于扩大有效的接受域,并基于低分辨率特征映射融合…

QT在VS环境中使用,控件显示中文乱码解决方法

首先来看乱码显示的效果如下&#xff1a; 上图左侧显示内容为中文&#xff0c;控件对应代码如下&#xff1a; QLabel* UserNameLabel new QLabel(QString("用户名&#xff1a;")); QLabel* NameLabel new QLabel(tr("姓名&#xff1a;"));下面我们对QL…

【Spine学习15】变换约束

变换约束&#xff1a;能让一个骨骼受另一个骨骼的变化影响。 1、选择m创建一个变换约束&#xff1a; 2、点击这个约束&#xff0c; 将移动数值拉的越满&#xff0c;m越接近s骨骼 当约束为0也就是默认的时候&#xff0c;m骨骼将不会受影响&#xff0c;变换约束可有可无。 tips…

物联网系统运维——实验备份与恢复,数据镜像软件DRBD介绍,DRBD的安装和应用,extundelete的安装和应用(重点),环境准备,配置设置

一.数据备份 1.数据备份的重要性 备份是系统中需要考虑的最重要的事项,虽然这在系统的整个规划,开发和测试过程中甚至占不到1%,看似不太重要且默默无闻的工作只有到恢复的时候才能真正体现出其重要性,任何数据的丢失与数据宕机&#xff0c;都是不可以被接收的。 2.数据备份策…

数据结构与算法笔记:高级篇 - 最短路径:地图软件是如何计算出最优出行路径的?

概述 基础篇的时候&#xff0c;我们学习了图的两种搜索算法&#xff0c;深度优先搜索和广度优先搜索。这两种算法主要是针对无权图的搜索算法。针对有权图&#xff0c;也就是图中的每一条变都有一个权重&#xff0c;我们该如何计算两点之间的最短路径&#xff08;经过的边的权…

ViT:3 Compact Architecture

大模型技术论文不断&#xff0c;每个月总会新增上千篇。本专栏精选论文重点解读&#xff0c;主题还是围绕着行业实践和工程量产。若在某个环节出现卡点&#xff0c;可以回到大模型必备腔调或者LLM背后的基础模型重新阅读。而最新科技&#xff08;Mamba,xLSTM,KAN&#xff09;则…

VMware共享文件夹设置

1、VMWare设置 1&#xff09;虚拟机 -> 设置 2&#xff09;点击 选项 -> 共享文件夹 -> 设置为 总是启用 -> 并 添加一个本地共享文件夹。设置完毕&#xff0c;点击 确定。 2、创建共享文件夹、挂载 1&#xff09;打开终端&#xff0c;在mnt/目录下创建一个名为…