Python抽象基类的定义与使用方法
作者:dongfanger 发布时间:2021-07-31 21:48:43
前言:
我们写Python
基本不需要自己创建抽象基类,而是通过鸭子类型来解决大部分问题。《流畅的Python》作者使用了15年Python
,但只在项目中创建过一个抽象基类。我们更多时候是创建现有抽象基类的子类,或者使用现有的抽象基类注册。本文的意义在于,了解抽象基类的定义与使用,可以帮助我们理解抽象基类是如何实现的,为我们以后学习后端语言(比如Java
、Golang
)打下基础。毕竟抽象基类是编程语言通用设计。
1、定义抽象基类的子类
先回顾下什么是抽象基类:Python
的抽象基类是指必须让继承它的子类去实现它所要求的抽象方法的类。
如下代码定义了抽象基类collections.MutableSequence的子类:
import collections
Card = collections.namedtuple('Card', ['rank', 'suit'])
class FrenchDeck2(collections.MutableSequence):
ranks = [str(n) for n in range(2, 11)] + list('JQKA')
suits = 'spades diamonds clubs hearts'.split()
def __init__(self):
self._cards = [Card(rank, suit) for suit in self.suits
for rank in self.ranks]
def __len__(self):
return len(self._cards)
def __getitem__(self, position):
return self._cards[position]
def __setitem__(self, position, value): # <1>
self._cards[position] = value
def __delitem__(self, position): # <2>
del self._cards[position]
def insert(self, position, value): # <3>
self._cards.insert(position, value)
通过抽象基类collections.MutableSequence
源码:
可以发现,它有三个抽象方法__setitem__
、 __delitem__
、insert
,所以FrenchDeck2
类必须实现它们。而对于其他非抽象方法比如append
、extend
、pop
等,则可以直接继承无需实现。
注意:Python
只会在运行时实例化FrenchDeck2
类时真正检查抽象方法的实现,如果未实现会抛出TypeError
异常,提示Can't instantiate abstract class
之类的。
2、标准库中的抽象基类
为了知道哪些抽象基类可以使用,我们可以看看标准库。
collections.abc
collections.abc
的抽象基类如下图所示:
Iterable、Container、Sized
这三个抽象基类是最基础的类,各个集合都继承了这三个抽象基类。
Itearble
通过__iter__
方法支持迭代Container
通过__contains__
方法支持in运算符Sized
通过__len__
方法支持len()
函数
Sequence、Mapping、Set
不可变集合类型,各自都有可变的子类。
MappingView
.items()
、.keys()
、 .values()
返回的对象分别是ItemsView
、KeysView
和ValuesView
的实例。
Callable、Hashable
为内置函数isinstance
提供支持,判断对象能不能调用或散列。
Iterator
迭代器。
numbers
numbers
的抽象基类有以下几种:
Number
Complex
Real
Rational
Integral
这叫做数字塔,顶部是超类,底部是子类。比如使用isinstance
(x, numbers.Integral)检查一个数是不是整数,这样代码就能接受int
、bool
(int的子类),再比如使用isinstance
(x, numbers.Real)检查浮点数,这样代码就能接受bool
、int
、float
、fractions.Fraction
。
3、定义抽象基类
抽象基类的示例代码如下:
# BEGIN TOMBOLA_ABC
import abc
class Tombola(abc.ABC): # <1>
@abc.abstractmethod
def load(self, iterable): # <2>
"""Add items from an iterable."""
@abc.abstractmethod
def pick(self): # <3>
"""Remove item at random, returning it.
This method should raise `LookupError` when the instance is empty.
"""
def loaded(self): # <4>
"""Return `True` if there's at least 1 item, `False` otherwise."""
return bool(self.inspect()) # <5>
def inspect(self):
"""Return a sorted tuple with the items currently inside."""
items = []
while True: # <6>
try:
items.append(self.pick())
except LookupError:
break
self.load(items) # <7>
return tuple(sorted(items))
# END TOMBOLA_ABC
要点:
继承
abc.ABC
使用
@abc.abstractmethod
装饰器标记抽象方法抽象基类也可以包含普通方法
抽象基类的子类必须覆盖抽象方法(普通方法可以不覆盖),可以使用super()函数调用抽象方法,为它添加功能,而不是从头开始实现
4、再看白鹅类型
白鹅类型的定义有一点难以理解,如果理解了虚拟子类,就能加快理解白鹅类型。虚拟子类并不是抽象基类的真正子类,而是注册到抽象基类上的子类,这样Python
就不会做强制检查了。
注册的方式有两种:
register方法
Python3.3
以前只能使用register方法,比如collections.abc
模块的源码中,把内置类型tuple
、str
、range
和memoryview
注册为Sequence
的虚拟子类:
Sequence.register(tuple)
Sequence.register(str)
Sequence.register(range)
Sequence.register(memoryview)
register装饰器
把TomboList
注册为Tombola
的虚拟子类:
@Tombola.register
class TomboList(list):
...
白鹅类型和鸭子类型是Python
的动态特性,它们的共同点是,只要长的像,Python
就不会做强制检查,鸭子类型是针对普通类的子类而言的,白鹅类型是针对抽象基类的虚拟子类而言的。
参考资料:
《流畅的Python》第11章 接口:从协议到抽象基类
来源:http://developer.51cto.com/art/202110/686024.htm
猜你喜欢
- 创建用户定义函数,它是返回值的已保存的 Transact-SQL 例程。用户定义函数不能用于执行一组修改全局数据库状态的操作。与系统函数一样
- 高阶函数英文叫Higher-order function。什么是高阶函数?我们以实际代码为例子,一步一步深入概念。变量可以指向函数以Pyth
- 最近做的asp.netMVC项目中需要对数据列表进行分类,这个本来就是基于bootstrap开发的后台,因此也就想着bootstrap是否有
- 字符串去除数字间的逗号在西文数字的表示中,很多格式是类似这样:123,456,789。如果得到这样的一个字符串,直接用int转换成整型肯定报
- 传统的网页BBS大多是采用CGI模式实现的,它的实现要求编程者既要掌握编程语言如Perl或C等,又要了解关于CGI模式的各项技术内容,因此要
- 假设你想找到本书中的某一个句子。你可以一页一页地逐页搜索,但这会花很多时间。而通过使用本书的索引,你可以很快地找到你要搜索的主题。表的索引与
- Python使用pandas导入xlsx格式的excel文件内容1. 基本导入在 Python中使用pandas导入.xlsx文件的方法是r
- 前言如何从图像中提取特征?第一次听说“特征提取”一词是在 YouTube 上的机器学习视频教程中,它
- modf()方法返回两个项的元组x的整数小数部分。这两个元组具有相同x符号。则返回一个浮点数的整数部分。语法以下是modf()方
- String Types(字符串类型)字符串类型Mysql支持多种字符串类型的变体。 这些数据类型在4.1和5.0版本中有较大的变化, 这使
- jquery基本入门 第一天:选择器相关 1.html()与.text() .html()取得第一个匹配元素的html内容。会带有标签,.t
- 有时候我们需要判断两个字符串内容是否相等,判断内容相等,我们用‘==',但是有时候发现print(str1)和print(str2)
- python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认
- 前几天,看到有人写了个superLink的东东,主要的做什么用呢?我们有时会给在大块元素加个window.location='htt
- 本文实例讲述了Python实现分割文件及合并文件的方法。分享给大家供大家参考。具体如下:分割文件split.py如下:#!/usr/bin/
- (需要安装psutil 用来获取服务器资源,以及pymongo驱动)#pip install psutil#pip install pymo
- Embedding词嵌入在 pytorch 中非常简单,只需要调用 torch.nn.Embedding(m, n) 就可以了,m 表示单词
- 今天看到某人博客推荐了http://dragoninteractive.com/这个网站,貌似一些效果做的比较不错,于是打开了看看,不过还真
- js 对url进行编码和解码三种编码和解码函数encodeURI和 decodeURI它着眼于对整个URL进行编码,因此除了常见的符号以外,
- 1,七层网络协议应表会传网数物:应用层、表示层、会话层: (这三层又可以合并为应用层,这样就是五层网络协议【osi五层协议】) python