谈谈Python中的接口与抽象基类

news/2024/12/14 14:28:03/文章来源:https://www.cnblogs.com/wang_yb/p/18606711

接触Python比较早的朋友可能都有这样的体会,Python语言虽然也支持面向对象的编程方式,

但是,不像那些纯面向对象的语言(比如Java.NET)那样严格和规范。

随着项目的规模逐步扩大之后,想要以一种清晰、可维护和可扩展的方式定义和实施对象的行为就变得越来越困难。

今天介绍的Python中两个为面向对象编程提供的强大工具:接口抽象基类

它们的英文分别为ProtocolsABC(Abstract Base Classes)。

ProtocolsPython3.8才开始引入的,有的地方也翻译成协议,我感觉翻译成接口更熟悉一些。

ABC引入的比较早,在Python3之后得到了改进和优化,现在和其他语言的抽象类相比,差别不大。

1. 接口(Protocols)

Python3.8开始在类型模块中引入的接口Protocols的概念,它提供了一种无需显式继承即可定义接口的方法。

接口Protocols定义了一组方法或属性,只要一个对象实现了这些方法或属性,就被视为满足该接口。

下面通过一个示例来帮助理解Protocols的使用,如果有面向对象编程的经验,很容易就能理解这个概念。

这个示例来自最近用的一个量化交易系统的一部分,这个功能需要从不同的来源获取数据,然后进行分析,最后将分析结果以不同的方式输出

这三个步骤(获取数据,分析和输出)中,

假设获取数据的来源有网络(API),文件(CSV)和数据库3种;

分析的步骤是统一的;输出的方式假设也有多种,比如邮件,短信等等。

根据这个描述,使用Protocols构建的获取数据和分析部分的代码如下:

输出的部分暂时不管

from typing import Protocol# 输入数据的接口
class InputData(Protocol):def get_data(self) -> str:passclass APIHandler:def get_data(self) -> str:print("get_data from API")return "get data from API"class CSVHandler:def get_data(self) -> str:print("get_data from CSV")return "get data from CSV"class SqliteHandler:def get_data(self) -> str:print("get_data from SQLITE DATABASE")return "get data from SQLITE DATABASE"# 分析数据
def analysis(i: InputData):data = i.get_data()print("开始处理数据...")

InputData继承了Protocol,其中定义了接口的函数get_data

只要实现了get_dataclass,比如APIHandlerCSVHandlerSqliteHandler,都可以当做InputData类型。

从代码可以看出,我们不需要用APIHandler去继承InputData,只要实现InputData中的所有方法就可以了。

这种灵活性确保了系统的可扩展性,我们可以添加新的数据源类型,而无需修改现有代码。

接下来我们测试上面的代码是否可以正常使用:

if __name__ == "__main__":i = APIHandler()analysis(i)print("\n")i = CSVHandler()analysis(i)print("\n")i = SqliteHandler()analysis(i)print("\n")

运行结果:

$  python.exe .\protocol_abc.py
get_data from API
开始处理数据...get_data from CSV
开始处理数据...get_data from SQLITE DATABASE
开始处理数据...

2. 抽象基类(ABC)

Protocol非常具有灵活性,但有时我们需要更结构化的方法,这就是抽象基类 (ABC) 的用武之地。

ABC 是一种通过定义子类必须实现的严格接口来强制执行一致行为的工具。

Protocol不同,ABC 需要显式继承,因此当我们希望在代码中明确定义层次结构时,ABC是更好的选择。

接着实现上一节示例中的输出部分,每种不同的输出需要不同的配置,

比如输出到邮件需要先配置邮箱的账号信息,输出到短信需要配置手机信息等等。

在这里,我们使用 ABC 来实现输出的基类。

# 输出的抽象基类
class OutputResult(ABC):def __init__(self):self.settings: dict = {}@abstractmethoddef send(self, data: str):pass@abstractmethoddef config(self, settings: dict):passclass OutputMail(ABC):def send(self, data: str):print(f"send {data} to {self.settings['name']}")def config(self, settings: dict):self.settings = settingsclass OutputMessage(ABC):def send(self, data: str):print(f"send {data} to {self.settings['name']}")def config(self, settings: dict):self.settings = settings

这里使用抽象基类的原因是输出时,并不是简单的调用send方法就可以的,还需要配置输出的参数,

所以用带有结构的抽象基类更好。

加上输出之后,上一节中的分析函数也改为:

# 分析数据
def analysis(i: InputData, o: OutputResult):data = i.get_data()print("开始处理数据...")data = data.replace("get data from ", "")o.send(data)

测试的代码如下:

if __name__ == "__main__":i = APIHandler()o = OutputMail()o.config({"name": "aaa@bbb.com"})analysis(i, o)print("\n")i = CSVHandler()o = OutputMessage()o.config({"name": "13911123456"})analysis(i, o)print("\n")i = SqliteHandler()o = OutputMail()o.config({"name": "xyz@www.com"})analysis(i, o)print("\n")

运行结果:

$  python.exe .\protocol_abc.py
get_data from API
开始处理数据...
send API to aaa@bbb.comget_data from CSV
开始处理数据...
send CSV to 13911123456get_data from SQLITE DATABASE
开始处理数据...
send SQLITE DATABASE to xyz@www.com

3. 两者的选择

当我们在实际的开发设计中,应该如何选择ProtocolABC呢?

其实,ProtocolABC之间的选择并不是非黑即白,这通常取决于项目的背景和你的目标。

一般来说,下面这些情况我们优先选择使用Protocol

  • 你正在使用现有代码或希望集成第三方库
  • 灵活性是首要任务,你不想强制执行严格的层次结构
  • 来自不相关的类层次结构的对象需要共享行为

而下面这些情况,优先选择ABC

  • 正在从头开始设计一个系统,并且需要加强结构
  • 类之间的关系是可预测的,并且继承是有意义的
  • 共享功能或默认行为可以减少重复并提高一致性

4. 总结

总的来说,ProtocolABC不是互相竞争的两种工具,它们是互补的。

我使用Protocol将类型安全改造到遗留系统中,而不需要大量重构。

另一方面,如果我在从头开始构建一个结构和一致性至关重要的系统时,会使用 ABC

在决定使用哪个时,请考虑项目的灵活性需求和长期目标。

Protocol提供灵活性和无缝集成,而 ABC 有助于建立结构和一致性。

通过了解它们各自的优势,你可以选择合适的方式来构建健壮、可维护的 Python 系统。

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

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

相关文章

集合的数据筛选

1.集合的数据筛选 1.创建集合,并封装数据2.许愿有一个方法,可以判断年龄,然后alt+回车创建该方法(忽略这里未封装数据)3.在方法中,创建新的集合,存储年龄小于18的对象4.遍历list集合,每一个元素都有可能小于18,再使用if判断,将满足条件的元素存入newlist集合5.在主方法…

转载:【AI系统】动手实现自动微分

在这章内容,会介绍是怎么实现自动微分的,因为代码量非常小,也许你也可以写一个玩玩。前面的文章当中,已经把自动微分的原理深入浅出的讲了一下,也引用了非常多的论文。有兴趣的可以顺着综述 A survey 这篇深扒一下。 前向自动微分原理 了解自动微分的不同实现方式非常有用…

基于.NET8+Vue3开发的权限管理个人博客系统

前言 今天大姚给大家分享一个基于.NET8+Vue3开发的权限管理&个人博客系统:Easy.Admin。 项目介绍 Easy.Admin是一个基于.NET8+Vue3+TypeScript开发的权限管理&个人博客系统,分为普通版本和SSR(服务端渲染,支持SEO),服务端渲染框架基于vite-plugin-ssr实现,并且支…

spring-boot-devtools 实现热部署

1.devtoolsspring为开发者提供了一个名为spring-boot-devtools的模块来使Spring Boot应用支持热部署,提高开发者的开发效率,无需手动重启Spring Boot应用。 2.项目搭建本文是采用IDEA搭建的Spring Boot应用,通过spring-boot-devtools配置,可以支持修改java文件会自动重启程…

IDEA bean json互转换插件

插件安装步骤:File->Settings->Plugins—>查找所需插件—>Install 或 File->Settings->Plugins—>Install plug from disk —>选择下载好的插件安装 一般插件安装后重启idea即可生效。 一、Java bean 转换 json 的插件 java-bean-to-json 下面详细安装…

转载:【AI系统】计算图的调度与执行

在前面的内容介绍过,深度学习的训练过程主要分为以下三个部分:1)前向计算、2)计算损失、3)更新权重参数。在训练神经网络时,前向传播和反向传播相互依赖。对于前向传播,沿着依赖的方向遍历计算图并计算其路径上的所有变量。然后将这些用于反向传播,其中计算顺序与计算图…

转载:【AI系统】微分实现方式

上一篇文章简单了解计算机中常用几种微分方式。本文将深入介绍 AI 框架离不开的核心功能:自动微分。 而自动微分则是分为前向微分和后向微分两种实现模式,不同的实现模式有不同的机制和计算逻辑,而无论哪种模式都离不开雅克比矩阵,所以我们也会深入了解一下雅克比矩阵的原理…

React16

React16免费基础视频教程 https://www.bilibili.com/video/BV1g4411i7po P1 01_React免费视频课程介绍 https://jspang.com 2019 5年前 react16 16.8.6 https://react.dev/ P2 02_React简介和Vue的对比 P3 03_React开发环境的搭建 npm i -g create-react-app@3.0.0 create-reac…

Input type (torch.cuda.FloatTensor) and weight type (torch.FloatTensor) should be the same

错误报在了forward里的Conv2d处。原因是函数写在forward里可能默认cpu,如果写在init构造函数里,就不需要再指定cuda。 修改为箭头指示就不再报错了。 【参考】 Input type (torch.cuda.FloatTensor) and weight type (torch.FloatTensor) should be the same-CSDN博客

jquery半透明拖拽窗口插件

这是一款jquery半透明拖拽窗口插件。该插件可以在页面生成可以拖拽、最大化、最小化的浮动窗口。在线演示 下载使用方法 在页面中引入style.css、jquery和jquery-translucent.js文件。<link rel="stylesheet" type="text/css" href="style.css&quo…

Marvelous Designer高版本更改界面字体大小

打开软件 打开 设置/用户自定义 - 用户自定义选择用户界面 - 显示 - 自动规模不勾选 - 分辨率选择大重启软件即可