Python 继承、多态、封装、抽象

面向对象编程(OOP)是 Python 中的一种重要编程范式,它通过类和对象来组织代码。OOP 的四个核心概念是继承(Inheritance)、多态(Polymorphism)、封装(Encapsulation)和数据抽象(Data Abstraction)。下面将详细介绍这四个概念。

继承(Inheritance)

继承是面向对象编程(OOP)的一个基本概念,它允许一个类(子类或派生类)继承另一个类(基类或父类)的属性和方法。继承促进了代码的重用,并有助于在类之间建立层次结构。在 Python 中,可以通过继承现有的基类来创建新的派生类。

继承的类型

1. 单继承(Single Inheritance)
  • 派生类从单个基类继承。

示例

class Animal:def __init__(self, name):self.name = namedef speak(self):passclass Dog(Animal):def speak(self):return f"{self.name} says Woof!"dog = Dog("Buddy")
print(dog.speak())  # 输出: Buddy says Woof!
2. 多继承(Multiple Inheritance)
  • 派生类从多个基类继承。

示例

class Animal:def __init__(self, name):self.name = nameclass Canine:def bark(self):return "Woof!"class Dog(Animal, Canine):def speak(self):return f"{self.name} says {self.bark()}"dog = Dog("Buddy")
print(dog.speak())  # 输出: Buddy says Woof!
3. 多级继承(Multilevel Inheritance)
  • 派生类从另一个派生类继承。

示例

class Animal:def __init__(self, name):self.name = nameclass Mammal(Animal):def __init__(self, name, has_fur):super().__init__(name)self.has_fur = has_furclass Dog(Mammal):def speak(self):return f"{self.name} says Woof!"dog = Dog("Buddy", True)
print(dog.speak())  # 输出: Buddy says Woof!
4. 层次继承(Hierarchical Inheritance)
  • 多个派生类从单个基类继承。

示例

class Animal:def __init__(self, name):self.name = nameclass Dog(Animal):def speak(self):return f"{self.name} says Woof!"class Cat(Animal):def speak(self):return f"{self.name} says Meow!"dog = Dog("Buddy")
cat = Cat("Whiskers")
print(dog.speak())  # 输出: Buddy says Woof!
print(cat.speak())  # 输出: Whiskers says Meow!
5. 混合继承(Hybrid Inheritance)
  • 两种或多种继承类型的组合。通常涉及层次继承、多级继承和多继承的混合。

示例

class Animal:def __init__(self, name):self.name = nameclass Canine(Animal):def bark(self):return "Woof!"class Feline(Animal):def meow(self):return "Meow!"class Dog(Canine):def speak(self):return f"{self.name} says {self.bark()}"class Cat(Feline):def speak(self):return f"{self.name} says {self.meow()}"dog = Dog("Buddy")
cat = Cat("Whiskers")
print(dog.speak())  # 输出: Buddy says Woof!
print(cat.speak())  # 输出: Whiskers says Meow!

详细解释

  1. 单继承

    • 最简单的继承形式,派生类从单个基类继承。
    • 适用于简单的类层次结构。
  2. 多继承

    • 派生类可以从多个基类继承。
    • 适用于需要从多个来源继承功能的场景。
    • 使用 super() 方法可以避免多重继承中的方法冲突。
  3. 多级继承

    • 派生类从另一个派生类继承,形成多级层次结构。
    • 适用于需要逐步细化功能的场景。
  4. 层次继承

    • 多个派生类从单个基类继承。
    • 适用于需要多个子类共享同一基类功能的场景。
  5. 混合继承

    • 两种或多种继承类型的组合。
    • 适用于复杂的类层次结构,需要灵活地组合多种继承方式。

注意事项

  • 方法解析顺序(MRO)

    • 在多继承中,Python 使用 C3 线性化算法来确定方法解析顺序(MRO)。
    • 可以使用 mro() 方法查看类的 MRO。
    • 例如:
      print(Dog.mro())
      
  • 使用 super()

    • super() 函数用于调用父类的方法,特别是在多继承中避免方法冲突。
    • 例如:
      class Mammal(Animal):def __init__(self, name, has_fur):super().__init__(name)self.has_fur = has_fur
      

多态(Polymorphism)

多态是面向对象编程(OOP)的一个核心概念,它允许不同类的对象被视为单一超类的成员。多态通过单一接口表示多种底层形式(数据类型),使得不同类可以提供同名但根据对象类具有不同行为的方法。多态通过方法重载和方法重写实现,促进了代码的灵活性和集成,使得编写通用、可重用的代码变得更加容易。

多态的基本概念

  1. 方法重写(Method Overriding)

    • 子类可以重写父类的方法,以提供不同的实现。
    • 通过方法重写,可以在子类中扩展或修改父类的行为。
  2. 方法重载(Method Overloading)

    • 同一个类中可以定义多个同名但参数不同的方法。
    • Python 本身不直接支持方法重载,但可以通过默认参数和可变参数等技术实现类似的效果。

示例

以下是一个展示多态的 Python 示例:

class Animal:def speak(self):passclass Dog(Animal):def speak(self):return "Woof!"class Cat(Animal):def speak(self):return "Meow!"def make_animal_speak(animal):print(animal.speak())dog = Dog()
cat = Cat()make_animal_speak(dog)  # 输出: Woof!
make_animal_speak(cat)  # 输出: Meow!

封装(Encapsulation)

封装是面向对象编程(OOP)的一个核心概念,它将方法(函数)和数据(属性)组织成一个类单元。封装还防止未经授权访问对象的某些部分,从而保护数据不被误用和无意干扰。通常,这是通过使用前导下划线或双下划线将某些属性或方法设为私有来实现的。封装通过公共方法提供受控访问,鼓励模块化并帮助维护对象数据的完整性。

封装的基本概念

  1. 私有属性和方法

    • 使用单个前导下划线 _ 表示“受保护”(protected)属性或方法,通常不应在类外部直接访问。
    • 使用双前导下划线 __ 表示“私有”(private)属性或方法,Python 会对其进行名称改写(name mangling),使其更难以在类外部访问。
  2. 公共方法

    • 提供公共方法来访问和修改私有属性,确保数据的完整性和安全性。

示例

以下是一个展示 Python 封装的示例:

class BankAccount:def __init__(self, owner, balance):self.owner = ownerself.__balance = balance  # 私有属性def deposit(self, amount):if amount > 0:self.__balance += amountprint(f"Deposited {amount}. New balance: {self.__balance}")else:print("Deposit amount must be positive.")def withdraw(self, amount):if 0 < amount <= self.__balance:self.__balance -= amountprint(f"Withdrew {amount}. New balance: {self.__balance}")else:print("Invalid withdrawal amount.")def get_balance(self):return self.__balance# 创建 BankAccount 类的对象
account = BankAccount("Alice", 1000)# 调用公共方法
account.deposit(500)  # 输出: Deposited 500. New balance: 1500
account.withdraw(200)  # 输出: Withdrew 200. New balance: 1300# 尝试直接访问私有属性(不推荐)
# print(account.__balance)  # 这将引发 AttributeError# 通过公共方法访问私有属性
print(account.get_balance())  # 输出: 1300

数据抽象(Data Abstraction)

数据抽象是面向对象编程(OOP)的一个重要原则,其目的是隐藏系统的复杂实现细节,只向用户展示必要和相关的信息。数据抽象通过提供简单和直观的接口,有助于提高效率和减少复杂性。它使程序员能够处理简化的系统表示,并管理复杂的操作。在 Python 中,可以使用抽象基类(Abstract Base Classes, ABC)和接口来实现数据抽象。

抽象基类(Abstract Base Classes, ABC)

抽象基类是定义了一组抽象方法的类,这些方法必须由派生类实现。抽象方法是没有具体实现的方法,只有方法签名。通过使用抽象基类,可以确保所有派生类都实现特定的方法,从而保证标准化和一致的交互。

示例

以下是一个使用抽象基类实现数据抽象的 Python 示例:

from abc import ABC, abstractmethodclass Shape(ABC):@abstractmethoddef area(self):pass@abstractmethoddef perimeter(self):passclass Rectangle(Shape):def __init__(self, width, height):self.width = widthself.height = heightdef area(self):return self.width * self.heightdef perimeter(self):return 2 * (self.width + self.height)class Circle(Shape):def __init__(self, radius):self.radius = radiusdef area(self):return 3.14 * self.radius ** 2def perimeter(self):return 2 * 3.14 * self.radius# 创建 Rectangle 和 Circle 对象
rect = Rectangle(10, 20)
circ = Circle(15)# 访问 area 和 perimeter 方法
print(f"Rectangle Area: {rect.area()}")       # 输出: Rectangle Area: 200
print(f"Rectangle Perimeter: {rect.perimeter()}")  # 输出: Rectangle Perimeter: 60
print(f"Circle Area: {circ.area()}")          # 输出: Circle Area: 706.5
print(f"Circle Perimeter: {circ.perimeter()}")     # 输出: Circle Perimeter: 94.2

详细解释

  1. 定义抽象基类 Shape

    • Shape 类继承自 ABC 类,并定义了两个抽象方法 areaperimeter
    • 抽象方法使用 @abstractmethod 装饰器声明,表示这些方法必须在派生类中实现。
  2. 定义具体类 RectangleCircle

    • Rectangle 类继承自 Shape 类,并实现了 areaperimeter 方法。
    • Circle 类继承自 Shape 类,并实现了 areaperimeter 方法。
  3. 创建对象并调用方法

    • 创建 RectangleCircle 对象。
    • 调用 areaperimeter 方法,计算并输出矩形和圆形的面积和周长。

数据抽象的好处

  1. 隐藏复杂性

    • 通过抽象基类,可以隐藏具体的实现细节,只向用户提供必要的接口。
    • 例如,用户不必了解如何计算矩形和圆形的面积和周长,只需调用相应的方法即可。
  2. 标准化接口

    • 抽象基类确保所有派生类都实现特定的方法,从而提供标准化的接口。
    • 例如,所有形状类都必须实现 areaperimeter 方法,这使得处理不同形状的代码更加一致和简洁。
  3. 提高可维护性

    • 通过数据抽象,代码更加模块化,易于维护和扩展。
    • 例如,如果需要添加新的形状类(如 Triangle),只需实现 areaperimeter 方法,而无需修改现有代码。
  4. 简化交互

    • 用户可以使用相同的接口与不同类型的对象进行交互,而不必了解每个对象的具体实现。
    • 例如,可以编写一个通用函数来处理任何 Shape 对象,而无需关心它是 Rectangle 还是 Circle

总结

面向对象编程的核心概念

  • 继承(Inheritance):允许一个类(子类)继承另一个类(父类)的属性和方法。
  • 多态(Polymorphism):允许不同类的对象被视为单一超类的成员,通过方法重写和重载实现。
  • 封装(Encapsulation):将数据和操作数据的方法绑定在一起,通过私有属性和公共方法保护数据。
  • 数据抽象(Data Abstraction):隐藏系统的复杂实现细节,只向用户展示必要和相关的信息,通过抽象基类和接口实现。

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

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

相关文章

02- 模块化编程-004 DB18B20温度数码显示

1、DB18B20采样与显示电路 2、DB18B20简介 DS18B20是一款高精度的单总线数字温度传感器&#xff0c;适用于多种温度测量场合。 温度范围&#xff1a;-55℃至125℃精度&#xff1a;可编程设定9至12位分辨率&#xff0c;对应温度分辨率0.5℃、0.25℃、0.125℃、0.0625℃通信&am…

[241102] Fedora Linux 41 正式发布 | Wasmer 5.0 发布

目录 Fedora Linux 41 正式发布Wasmer 5.0 发布 Fedora Linux 41 正式发布 主要更新&#xff1a; DNF 5: 更快、更小、依赖更少&#xff0c;统一了容器、服务器、桌面和设备的包管理体验。桌面更新&#xff1a; Fedora Workstation 41 基于 GNOME 47&#xff0c;默认终端改为…

C++继承

文章目录 一、继承的概念和定义1、继承的概念2、继承的定义3、继承基类成员访问方式的变化 二、基类和派生类之间的转换三、继承中的作用域1、隐藏规则 四、派生类的默认成员函数1、常见默认成员函数2、实现一个不能被继承的类 五、继承与友元六、继承与静态成员变量七、多继承…

嵌入式linux系统中串口驱动框架分析

大家好,今天主要给大家分享一下,如何使用linux系统中的串口实现。 第一:串口基本简介 串口是很常见的一个外设,在Linux下通常通过串口和其他设备或传感器进行通信。根据电平的不同,串口可以分为TTL和RS232。不管是什么样的电平接口,驱动程序是一样的。 第二:Linux下UAR…

秋日盛景,北京马拉松万人齐跑,秀域人工智能理疗获好评无数

在金秋送爽的北京&#xff0c;一场全民瞩目的体育盛事——北京马拉松顺利开跑&#xff0c;再次点燃了这座城市的运动激情。各地跑者齐聚双奥之城&#xff0c;共同奔赴在秋日美景之中。      回首往昔&#xff0c;1981年9月27日&#xff0c;首届北京马拉松赛的举办&#xff…

Java-I/O框架09:InputStreamReader、OutputStreamWriter使用

视频链接&#xff1a;16.24 转换流的使用_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1Tz4y1X7H7?spm_id_from333.788.videopod.episodes&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b5&p24 1.InputStreamReader使用 package com.yundait.Demo05;import java…

JavaScript语法基础(JS使用方式, 输出方式, 变量,数据类型,运算符,程序控制语句)

目录 JS概念 一.JS使用方式 &#xff08;1&#xff09;行内式 &#xff08;2&#xff09;内嵌式 &#xff08;3&#xff09;外链式 二.JS输出方式 三.基本语法 1.变量 &#xff08;1&#xff09;变量的命名 &#xff08;2&#xff09;变量的声明 &#xff08;3&#xf…

NPOI 操作详解(操作Excel)

目录 1. 安装 NPOI 2. 使用 NPOI 创建新 Excel 文件 3. 设置列宽和行高 1. 设置列宽 2. 设置行高 3. 同时设置列宽和行高 4. 设置统一的行高 5. 设置统一的列宽 6. 应用统一的行高和列宽 4. 合并单元格 5. 设置单元格样式&#xff08;字体、边框、背景色等&#xf…

codigger体验过程记录

一、codigger的介绍 codigger是一款分布式操作系统&#xff0c;主要是便于进行个人移动化办公、在线协同办公、开发和娱乐的私人应用。 收到codigger开发者朋友的邀请&#xff0c;于是进行了这款产品的升级体验&#xff0c;接下来就简单介绍一下自己的体验过程吧。 二、登录系…

使用labelme中的AI模型提升数据标注速度

1、打开labelme 2、选择AI模型 选择之后&#xff0c;会自动进行模型下载&#xff0c;推荐开梯子&#xff0c;否则可能下载失败&#xff0c;最小的EfficientSam 在30M左右 3、右键选择AI多边形&#xff08;分割&#xff09;或AI蒙版&#xff08;目标检测&#xff09;&#xf…

全双工通信协议WebSocket——使用WebSocket实现智能学习助手/聊天室功能

一.什么是WebSocket&#xff1f; WebSocket是基于TCP的一种新的网络协议。它实现了浏览器与服务器的全双工通信——浏览器和服务器只需要完成一次握手&#xff0c;两者之间就可以创建持久性的连接&#xff0c;并进行双向数据传输 HTTP 协议是一种无状态的、无连接的、单向的应用…

音乐网站新篇章:SpringBoot Web实现

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

【初阶数据结构篇】链式结构二叉树(二叉链)的实现(感受递归暴力美学)

文章目录 须知 &#x1f4ac; 欢迎讨论&#xff1a;如果你在学习过程中有任何问题或想法&#xff0c;欢迎在评论区留言&#xff0c;我们一起交流学习。你的支持是我继续创作的动力&#xff01; &#x1f44d; 点赞、收藏与分享&#xff1a;觉得这篇文章对你有帮助吗&#xff1…

el-talble selection行 初始默认勾选

导言 el-talble selection 行&#xff08;选择列&#xff09;用于显示复选框&#xff0c;让用户可以选择或取消选择某些表格行&#xff0c;常用于批量操作场景。 刚刚试了下&#xff0c;想加深印象记录一下当学习碎片。参考的是表格多选并根据每行值初始化选中状态&#xff08;…

RabbitMQ交换机类型

RabbitMQ交换机类型 1、RabbitMQ工作模型2、RabbitMQ交换机类型2.1、Fanout Exchange&#xff08;扇形&#xff09;2.1.1、介绍2.1.2、示例2.1.2.1、生产者2.1.2.2、消费者2.1.2.3、测试 2.2、Direct Exchange&#xff08;直连&#xff09;2.2.1、介绍2.2.2、示例2.2.2.1、生产…

数据结构---排序(上)

一.直接插入排序 思想&#xff1a;将一个个未排序的数字插入到已经排好顺序的数组中。 例如&#xff1a; 思路&#xff1a;先将前两个数字排序&#xff0c;然后将后面数字与前面数字比较排序。 操作&#xff1a; 1.引入变量 i 遍历数组[1&#xff0c;array.lenth] 2.用临时…

ai翻唱部分步骤

模型部署 我是用的RVC进行的训练&#xff0c;也可以使用so-vits-svc。 通过百度网盘分享的文件&#xff1a;RVC-beta 链接&#xff1a;https://pan.baidu.com/s/1c99jR2fLChoqUFqf9gLUzg 提取码&#xff1a;4090 以Nvida显卡为例&#xff0c;分别下载“RVC1006Nvidia”和…

C++的stack和Queue

1.简单实现stack 构建一个模板&#xff0c;俩个参数&#xff0c;这里第一个一般是数据的类型&#xff0c;第二个是由什么来实现栈&#xff0c;在主函数里传了int和vector<int>&#xff0c;第二个不传参也可以&#xff0c;因为是缺省参数&#xff0c;默认为vector&#x…

默认路由:实现内网所有网段流量走一条默认路由访问外网

默认路由 Tip&#xff1a;默认路由一般指出口网关设备的出口路由。实现所有网段流量都走一条路由。 实验模拟&#xff1a;公司内部pc 通过出口网关 访问运营商内部 baidu服务 isp网关配置&#xff1a; <Huawei>sy Enter system view, return user view with CtrlZ. …

蘑菇书(EasyRL)学习笔记(2)

1、序列决策 1.1、智能体和环境 如下图所示&#xff0c;序列决策过程是智能体与环境之间的交互&#xff0c;智能体通过动作影响环境&#xff0c;环境则返回观测和奖励。智能体的目标是从这些反馈中学习出能最大化长期奖励的策略&#xff0c;这一过程通过不断试错和调整实现强化…