Python抽象基类的定义与使用方法

 更新时间:2022年04月26日 09:02:46   作者:dongfanger  
这篇文章主要介绍了Python抽象基类的定义与使用方法,Python的抽象基类是指必须让继承它的子类去实现它所要求的抽象方法的类,下面文章内容将详细介绍相关资料,需要的朋友可以参考一下

前言:

我们写Python基本不需要自己创建抽象基类,而是通过鸭子类型来解决大部分问题。《流畅的Python》作者使用了15年Python,但只在项目中创建过一个抽象基类。我们更多时候是创建现有抽象基类的子类,或者使用现有的抽象基类注册。本文的意义在于,了解抽象基类的定义与使用,可以帮助我们理解抽象基类是如何实现的,为我们以后学习后端语言(比如JavaGolang)打下基础。毕竟抽象基类是编程语言通用设计。

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类必须实现它们。而对于其他非抽象方法比如appendextendpop等,则可以直接继承无需实现。

注意: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()返回的对象分别是ItemsViewKeysViewValuesView的实例。

Callable、Hashable

为内置函数isinstance提供支持,判断对象能不能调用或散列。

Iterator

迭代器。

numbers

numbers的抽象基类有以下几种:

  • Number
  • Complex
  • Real
  • Rational
  • Integral

这叫做数字塔,顶部是超类,底部是子类。比如使用isinstance(x, numbers.Integral)检查一个数是不是整数,这样代码就能接受intbool(int的子类),再比如使用isinstance(x, numbers.Real)检查浮点数,这样代码就能接受boolintfloatfractions.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模块的源码中,把内置类型tuplestrrangememoryview注册为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章 接口:从协议到抽象基类

到此这篇关于Python抽象基类的定义与使用方法的文章就介绍到这了,更多相关Python抽象基类的定义与使用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • python进程间数据交互的几种实现方式

    python进程间数据交互的几种实现方式

    本文主要介绍了python进程数据交互的几种实现方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • Python中json.dumps()和json.dump()的区别小结

    Python中json.dumps()和json.dump()的区别小结

    在Python中,json.dumps()和json.dump()是两个常用的函数,本文主要介绍了Python中json.dumps()和json.dump()的区别小结,具有一定的参考价值,感兴趣的可以了解一下
    2024-02-02
  • Python基于easygui实现pdf和word转换小程序

    Python基于easygui实现pdf和word转换小程序

    这篇文章主要为大家详细介绍了Python如何基于easygui实现pdf和word转换小程序,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-04-04
  • pandas 实现 in 和 not in 的用法及使用心得

    pandas 实现 in 和 not in 的用法及使用心得

    pandas按条件筛选数据时,除了使用query()方法,还可以使用isin和对isin取反进行条件筛选,今天通过本文给大家介绍pandas 实现 in 和 not in 的用法及使用心得,感兴趣的朋友跟随小编一起看看吧
    2023-01-01
  • Python实现葵花8号卫星数据自动下载实例

    Python实现葵花8号卫星数据自动下载实例

    这篇文章主要为大家介绍了Python实现葵花8号卫星数据自动下载实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10
  • python输入多行的方法总结

    python输入多行的方法总结

    Python中的Input()函数在输入时,遇到回车符,那么一次输入就结束了,这不能满足输入多行文本并且行数也不确定的情形,当然输入空行也是允许的,本文给大家总结了python输入多行的方法,需要的朋友可以参考下
    2024-04-04
  • python sys模块使用方法介绍

    python sys模块使用方法介绍

    sys模块是最常用的和python解释器交互的模块,sys模块可供访问由解释器(interpreter)使用或维护的变量和与解释器进行交互的函数,需要的朋友可以参考下
    2022-08-08
  • 关于tensorflow的几种参数初始化方法小结

    关于tensorflow的几种参数初始化方法小结

    今天小编就为大家分享一篇关于tensorflow的几种参数初始化方法小结,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-01-01
  • python实现将中文日期转换为数字日期

    python实现将中文日期转换为数字日期

    这篇文章主要介绍了python实现将中文日期转换为数字日期,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07
  • 详解Python3.8+PyQt5+pyqt5-tools+Pycharm配置详细教程

    详解Python3.8+PyQt5+pyqt5-tools+Pycharm配置详细教程

    这篇文章主要介绍了Python3.8+PyQt5+pyqt5-tools+Pycharm配置详细教程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2020-11-11

最新评论