gpt4 book ai didi

Python抽象基类的定义与使用

转载 作者:qq735679552 更新时间:2022-09-29 22:32:09 33 4
gpt4 key购买 nike

CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.

这篇CFSDN的博客文章Python抽象基类的定义与使用由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.

Python抽象基类的定义与使用

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

定义抽象基类的子类

先回顾下什么是抽象基类:Python的抽象基类是指必须让继承它的子类去实现它所要求的抽象方法的类。如下代码定义了抽象基类collections.MutableSequence的子类:

  1. importcollections
  2.  
  3. Card=collections.namedtuple('Card',['rank','suit'])
  4.  
  5. classFrenchDeck2(collections.MutableSequence):
  6. ranks=[str(n)forninrange(2,11)]+list('JQKA')
  7. suits='spadesdiamondsclubshearts'.split()
  8.  
  9. def__init__(self):
  10. self._cards=[Card(rank,suit)forsuitinself.suits
  11. forrankinself.ranks]
  12.  
  13. def__len__(self):
  14. returnlen(self._cards)
  15.  
  16. def__getitem__(self,position):
  17. returnself._cards[position]
  18.  
  19. def__setitem__(self,position,value):#<1>
  20. self._cards[position]=value
  21.  
  22. def__delitem__(self,position):#<2>
  23. delself._cards[position]
  24.  
  25. definsert(self,position,value):#<3>
  26. self._cards.insert(position,value)

通过抽象基类collections.MutableSequence源码:

Python抽象基类的定义与使用

可以发现,它有三个抽象方法__setitem__、__delitem__、insert,所以FrenchDeck2类必须实现它们。而对于其他非抽象方法比如append、extend、pop等,则可以直接继承无需实现.

注意,Python只会在运行时实例化FrenchDeck2类时真正检查抽象方法的实现,如果未实现会抛出TypeError异常,提示Can't instantiate abstract class之类的.

标准库中的抽象基类

为了知道哪些抽象基类可以使用,我们可以看看标准库.

collections.abc

collections.abc的抽象基类如下图所示:

Python抽象基类的定义与使用

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.

定义抽象基类

本小结可以跳过。不过了解抽象基类的定义有助于阅读标准库和其他包中的抽象基类源码.

抽象基类的示例代码如下:

  1. #BEGINTOMBOLA_ABC
  2.  
  3. importabc
  4.  
  5. classTombola(abc.ABC):#<1>
  6.  
  7. @abc.abstractmethod
  8. defload(self,iterable):#<2>
  9. """Additemsfromaniterable."""
  10.  
  11. @abc.abstractmethod
  12. defpick(self):#<3>
  13. """Removeitematrandom,returningit.
  14. Thismethodshouldraise`LookupError`whentheinstanceisempty.
  15. """
  16.  
  17. defloaded(self):#<4>
  18. """Return`True`ifthere'satleast1item,`False`otherwise."""
  19. returnbool(self.inspect())#<5>
  20.  
  21.  
  22. definspect(self):
  23. """Returnasortedtuplewiththeitemscurrentlyinside."""
  24. items=[]
  25. whileTrue:#<6>
  26. try:
  27. items.append(self.pick())
  28. exceptLookupError:
  29. break
  30. self.load(items)#<7>
  31. returntuple(sorted(items))
  32.  
  33.  
  34. #ENDTOMBOLA_ABC

要点:

  1. 继承abc.ABC
  2. 使用@abc.abstractmethod装饰器标记抽象方法
  3. 抽象基类也可以包含普通方法
  4. 抽象基类的子类必须覆盖抽象方法(普通方法可以不覆盖),可以使用super()函数调用抽象方法,为它添加功能,而不是从头开始实现

再看白鹅类型

白鹅类型的定义有一点难以理解,如果理解了虚拟子类,就能加快理解白鹅类型。虚拟子类并不是抽象基类的真正子类,而是注册到抽象基类上的子类,这样Python就不会做强制检查了.

注册的方式有两种:

register方法 。

Python3.3以前只能使用register方法,比如collections.abc模块的源码中,把内置类型tuple、str、range和memoryview注册为Sequence的虚拟子类:

  1. Sequence.register(tuple)
  2. Sequence.register(str)
  3. Sequence.register(range)
  4. Sequence.register(memoryview)

register装饰器 。

把TomboList注册为Tombola的虚拟子类:

  1. @Tombola.register
  2. classTomboList(list):
  3. ...

白鹅类型和鸭子类型是Python的动态特性,它们的共同点是,只要长的像,Python就不会做强制检查,鸭子类型是针对普通类的子类而言的,白鹅类型是针对抽象基类的虚拟子类而言的.

参考资料:

《流畅的Python》第11章 接口:从协议到抽象基类 。

原文链接:https://mp.weixin.qq.com/s/papFgSyN9Iqofu5faJDT6g 。

最后此篇关于Python抽象基类的定义与使用的文章就讲到这里了,如果你想了解更多关于Python抽象基类的定义与使用的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com