Python 抽象基类 ABC :从实践到优雅
今天我们来聊聊 Python 中的抽象基类(Abstract Base Class,简称 ABC)。虽然这个概念在 Python 中已经存在很久了,但在日常开发中,很多人可能用得并不多,或者用得不够优雅。
让我们从一个实际场景开始:假设你正在开发一个文件处理系统,需要支持不同格式的文件读写,比如 JSON、CSV、XML 等。
初始版本:简单但不够严谨
我们先来看看最简单的实现方式:
class FileHandler:
def read(self, filename):
pass
def write(self, filename, data):
pass
class JsonHandler(FileHandler):
def read(self, filename):
import json
with open(filename, 'r') as f:
return json.load(f)
def write(self, filename, data):
import json
with open(filename, 'w') as f:
json.dump(data, f)
class CsvHandler(FileHandler):
def read(self, filename):
import csv
with open(filename, 'r') as f:
return list(csv.reader(f))
这个实现看起来没什么问题,但实际上存在几个隐患:
- 无法强制子类实现所有必要的方法
- 基类方法的签名(参数列表)可能与子类不一致
- 没有明确的接口契约
改进版本:使用抽象基类
让我们引入 abc.ABC 来改进这个设计:
from abc import ABC, abstractmethod
class FileHandler(ABC):
@abstractmethod
def read(self, filename: str):
"""读取文件内容"""
pass
@abstractmethod
def write(self, filename: str, data: any):
"""写入文件内容"""
pass
class JsonHandler(FileHandler):
def read(self, filename: str):
import json
with open(filename, 'r') as f:
return json.load(f)
def write(self, filename: str, data: any):
import json
with open(filename, 'w') as f:
json.dump(data, f)
这个版本引入了两个重要的改进:
- 使用
ABC将FileHandler声明为抽象基类 - 使用
@abstractmethod装饰器标记抽象方法
现在,如果我们尝试实例化一个没有实现所有抽象方法的子类,Python 会抛出异常:
# 这个类缺少 write 方法的实现
class BrokenHandler(FileHandler):
def read(self, filename: str):
return "some data"
# 这行代码会抛出 TypeError
handler = BrokenHandler() # TypeError: Can't instantiate abstract class BrokenHandler with abstract method write
进一步优化:添加类型提示和接口约束
让我们再进一步,添加类型提示和更严格的接口约束:
from abc import ABC, abstractmethod
from typing import Any, List, Dict, Union
class FileHandler(ABC):
@abstractmethod
def read(self, filename: str) -> Union[Dict, List]:
"""读取文件内容并返回解析后的数据结构"""
pass
@abstractmethod
def write(self, filename: str, data: Union[Dict, List]) -> None:
"""将数据结构写入文件"""
pass
@property
@abstractmethod
def supported_extensions(self) -> List[str]:
"""返回支持的文件扩展名列表"""
pass
class JsonHandler(FileHandler):
def read(self, filename: str) -> Dict:
import json
with open(filename, 'r') as f:
return json.load(f)
def write(self, filename: str, data: Dict) -> None:
import json
with open(filename, 'w') as f:
json.dump(data, f)
@property
def supported_extensions(self) -> List[str]:
return ['.json']
# 使用示例
def process_file(handler: FileHandler, filename: str) -> None:
if any(filename.endswith(ext) for ext in handler.supported_extensions):
data = handler.read(filename)
# 处理数据...
handler.write(f'processed_{filename}', data)
else:
raise ValueError(f"Unsupported file extension for {filename}")
这个最终版本的改进包括:
- 添加了类型提示,提高代码的可读性和可维护性
- 引入了抽象属性(
supported_extensions),使接口更完整 - 通过
Union类型提供了更灵活的数据类型支持 - 提供了清晰的文档字符串
使用抽象基类的好处
接口契约:抽象基类提供了明确的接口定义,任何违反契约的实现都会在运行前被发现。
代码可读性:通过抽象方法清晰地表明了子类需要实现的功能。
类型安全:结合类型提示,我们可以在开发时就发现潜在的类型错误。
设计模式支持:抽象基类非常适合实现诸如工厂模式、策略模式等设计模式。
NotImplementedError 还是 ABC?
很多 Python 开发者会使用 NotImplementedError 来标记需要子类实现的方法:
class FileHandler:
def read(self, filename: str) -> Dict:
raise NotImplementedError("Subclass must implement read method")
def write(self, filename: str, data: Dict) -> None:
raise NotImplementedError("Subclass must implement write method")
这种方式看起来也能达到目的,但与 ABC 相比有几个明显的劣势:
- 延迟检查:使用
NotImplementedError只能在运行时发现问题,而 ABC 在实例化时就会检查。
# 使用 NotImplementedError 的情况
class BadHandler(FileHandler):
pass
handler = BadHandler() # 这行代码可以执行
handler.read("test.txt") # 直到这里才会报错
# 使用 ABC 的情况
class BadHandler(FileHandler): # FileHandler 是 ABC
pass
handler = BadHandler() # 直接在这里就会报错
缺乏语义:
NotImplementedError本质上是一个异常,而不是一个接口契约。IDE 支持:现代 IDE 对 ABC 的支持更好,能提供更准确的代码提示和检查。
不过,NotImplementedError 在某些场景下仍然有其价值:
- 当你想在基类中提供部分实现,但某些方法必须由子类覆盖时:
from abc import ABC, abstractmethod
class FileHandler(ABC):
@abstractmethod
def read(self, filename: str) -> Dict:
pass
def process(self, filename: str) -> Dict:
data = self.read(filename)
if not self._validate(data):
raise ValueError("Invalid data format")
return self._transform(data)
def _validate(self, data: Dict) -> bool:
raise NotImplementedError("Subclass should implement validation")
def _transform(self, data: Dict) -> Dict:
# 默认实现
return data
这里,_validate 使用 NotImplementedError 而不是 @abstractmethod,表明它是一个可选的扩展点,而不是必须实现的接口。
代码检查工具的配合
主流的 Python 代码检查工具(pylint、flake8)都对抽象基类提供了良好的支持。
Pylint
Pylint 可以检测到未实现的抽象方法:
# pylint: disable=missing-module-docstring
from abc import ABC, abstractmethod
class Base(ABC):
@abstractmethod
def foo(self):
pass
class Derived(Base): # pylint: error: Abstract method 'foo' not implemented
pass
你可以在 .pylintrc 中配置相关规则:
[MESSAGES CONTROL]
# 启用抽象类检查
enable=abstract-method
Flake8
Flake8 本身不直接检查抽象方法实现,但可以通过插件增强这个能力:
pip install flake8-abstract-base-class
配置 .flake8:
[flake8]
max-complexity = 10
extend-ignore = ABC001
metaclass=ABCMeta vs ABC
在 Python 中,有两种方式定义抽象基类:
# 方式 1:直接继承 ABC
from abc import ABC, abstractmethod
class FileHandler(ABC):
@abstractmethod
def read(self):
pass
# 方式 2:使用 metaclass
from abc import ABCMeta, abstractmethod
class FileHandler(metaclass=ABCMeta):
@abstractmethod
def read(self):
pass
这两种方式在功能上是等价的,因为 ABC 类本身就是用 ABCMeta 作为元类定义的:
class ABC(metaclass=ABCMeta):
"""Helper class that provides a standard way to create an ABC using
inheritance.
"""
pass
选择建议:
推荐使用 ABC:
- 代码更简洁
- 更符合 Python 的简单直观原则
- 是 Python 3.4+ 后推荐的方式
使用 metaclass=ABCMeta 的场景:
- 当你的类已经有其他元类时
- 需要自定义元类行为时
例如,当你需要组合多个元类的功能时:
class MyMeta(type):
def __new__(cls, name, bases, namespace):
# 自定义的元类行为
return super().__new__(cls, name, bases, namespace)
class CombinedMeta(ABCMeta, MyMeta):
pass
class MyHandler(metaclass=CombinedMeta):
@abstractmethod
def handle(self):
pass
实践建议
当你需要确保一组类遵循相同的接口时,使用抽象基类。
优先使用类型提示,它们能帮助开发者更好地理解代码。
适当使用抽象属性(
@property+@abstractmethod),它们也是接口的重要组成部分。在文档字符串中清晰地说明方法的预期行为和返回值。
通过这个实例,我们可以看到抽象基类如何帮助我们写出更加健壮和优雅的 Python 代码。它不仅能够捕获接口违规,还能提供更好的代码提示和文档支持。在下一个项目中,不妨试试用抽象基类来设计你的接口!
Python 抽象基类 ABC :从实践到优雅的更多相关文章
- Python抽象基类:ABC谢谢你,因为有你,温暖了四季!
Python抽象基类:ABC谢谢你,因为有你,温暖了四季! Python抽象基类:ABC谢谢你,因为有你,温暖了四季! 实例方法.类方法和静态方法 抽象类 具名元组 参考资料 最近阅读了<Pyt ...
- Python抽象基类之声明协议
抽象基类之--声明协议 上回讲了Python中抽象基类的大概,相信大家对abcmeta以及什么是抽象基类已经有所了解.传送门 现在我们来讲讲抽象基类的另一个常用用法--声明协议 所谓声明协议,有点像J ...
- python抽象基类
抽象基类 抽象基类提了一种方式,用以组织对象的层次结构,做出关于所需方法的断言,以及实现其他一些功能 要定义抽象基类,需要使用abc模块,该模块定义了一个元类(ABCMeta) 和一组装饰器(@abs ...
- 抽象基类(ABC),纯虚函数
#ifndef _ACCTABC_H_ #define _ACCTABC_H_ //(* #include <iostream> #include <string> //*) ...
- guxh的python笔记七:抽象基类
1,鸭子类型和白鹅类型 1.1,白鹅类型 白鹅类型对接口有明确定义,比如不可变序列(Sequence),需要实现__contains__,__iter__,__len__,__getitem__,__ ...
- Python高级主题:Python ABC(抽象基类)
#抽象类实例 作用统一规范接口,降低使用复杂度.import abcclass Animal(metaclass = abc.ABCMeta): ##只能被继承,不能实例化,实例化会报错 @abc.a ...
- 【Python】【元编程】【从协议到抽象基类】
"""class Vector2d: typecode = 'd' def __init__(self,x,y): self.__x = float(x) self.__ ...
- python面对对象编程---------6:抽象基类
抽象基本类的几大特点: 1:要定义但是并不完整的实现所有方法 2:基本的意思是作为父类 3:父类需要明确表示出那些方法的特征,这样在写子类时更加简单明白 用抽象基本类的地方: 1:用作父类 2:用作检 ...
- 流畅python学习笔记:第十一章:抽象基类
__getitem__实现可迭代对象.要将一个对象变成一个可迭代的对象,通常都要实现__iter__.但是如果没有__iter__的话,实现了__getitem__也可以实现迭代.我们还是用第一章扑克 ...
- Python 接口:从协议到抽象基类
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px Helvetica } 抽象基类的常见用途:实现接口时作为超类使用.然后,说明抽象基类如何检查 ...
随机推荐
- 云原生周刊 | 人类、机器人与 Kubernetes
近日 Grafana 官网发表了一篇博客介绍了 2022 年比较有意思.脑洞大开的一些 Grafana 使用案例,比如监控特斯拉 Model 3 的充电状态.OTA 更新状况等等. 海事技术供应商 R ...
- 几行代码带你用TinyEngine低代码引擎开发侧边栏插件
本文分享自华为云社区<实操上手TinyEngine低代码引擎插件化开发>,作者:OpenTiny. 1.背景介绍 1.1 TinyEngine 低代码引擎简介 低代码开发是近些年非常热门的 ...
- MISC 高手进阶区 1-5
1.reverseME 题目描述 无 附件 一个 .jpg 的图片 是一个flag字符串的图片镜像. reverse V-T If you reverse the order of a set of ...
- Linux基础-学会使用命令帮助
概述 使用 whatis 使用 man 查看命令程序路径 which 总结 参考资料 概述 Linux 命令及其参数繁多,大多数人都是无法记住全部功能和具体参数意思的.在 linux 终端,面对命令不 ...
- "(UE4Editor.exe中)处有未经处理的异常:0xC0000005:读取位置0x0000000000000000时发生访问冲突"报错情况+解决方法+原因分析
报错情况:使用ue4.27 Slate编写Widget时想通过获取Worl(通过本地PlayerController获取)来实现"设置定时任务为在音乐结束后自动触发函数"的功能 p ...
- 大便系统怎样安装RPM包
alien包转换工具 如果我们有很喜欢的RPM包,而又没有deb版本. 怎么办~? 可以同过alien来转换或者直接安装,这个小家伙可是个很方便的东西! 基本命令如下: 首先通过apt-get ins ...
- 从0搭建一个FIFO模块-02(系统架构)
一.异步FIFO需要注意的问题 所谓异步FIFO,指的是写时钟与读时钟可以不同步,读时钟可以比写时钟快,反之亦然.思考一下,这样会直接地造成两个问题: 由于异步FIFO的基本存储单元是双端口RAM,因 ...
- python爬虫 正则表达式详解
正则表达式 最近学校布置了一个关于python爬虫的期末作业,而我之前对python爬虫一直都比较感兴趣但是没有系统的学过,就想借此机会开个新坑来系统学习和应用python爬虫,那我们开始吧 正则表达 ...
- 迁移到 Eclipse: Eclipse 对 IntelliJ IDEAA 评估开发指南
为何考虑 Eclipse 以及它与 IntelliJ IDEA 有什么不同 Eclipse 是一个免费的.正日益流行起来的 Java 集成开发环境,最新版本的 Eclipse 中提供了很多特性,这些特 ...
- vue之vuex使用
如图所示,它是一个程序里面的状态管理模式,它是集中式存储所有组件的状态的小仓库,并且保持我们存储的状态以一种可以预测的方式发生变化.对于可以预测,现在我不多做说明,相信在看完这篇文章之后,你就会有自己 ...