成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python抽象基類的定義與使用

開發 后端
我們寫Python基本不需要自己創建抽象基類,而是通過鴨子類型來解決大部分問題。《流暢的Python》作者使用了15年Python,但只在項目中創建過一個抽象基類。

[[429320]]

我們寫Python基本不需要自己創建抽象基類,而是通過鴨子類型來解決大部分問題。《流暢的Python》作者使用了15年Python,但只在項目中創建過一個抽象基類。我們更多時候是創建現有抽象基類的子類,或者使用現有的抽象基類注冊。本文的意義在于,了解抽象基類的定義與使用,可以幫助我們理解抽象基類是如何實現的,為我們以后學習后端語言(比如Java、Golang)打下基礎。畢竟抽象基類是編程語言通用設計。

定義抽象基類的子類

先回顧下什么是抽象基類:Python的抽象基類是指必須讓繼承它的子類去實現它所要求的抽象方法的類。如下代碼定義了抽象基類collections.MutableSequence的子類:

  1. import collections 
  2.  
  3. Card = collections.namedtuple('Card', ['rank''suit']) 
  4.  
  5. class FrenchDeck2(collections.MutableSequence): 
  6.     ranks = [str(n) for n in range(2, 11)] + list('JQKA'
  7.     suits = 'spades diamonds clubs hearts'.split() 
  8.  
  9.     def __init__(self): 
  10.         self._cards = [Card(rank, suit) for suit in self.suits 
  11.                                         for rank in self.ranks] 
  12.  
  13.     def __len__(self): 
  14.         return len(self._cards) 
  15.  
  16.     def __getitem__(self, position): 
  17.         return self._cards[position] 
  18.  
  19.     def __setitem__(self, position, value):  # <1> 
  20.         self._cards[position] = value 
  21.  
  22.     def __delitem__(self, position):  # <2> 
  23.         del self._cards[position] 
  24.  
  25.     def insert(self, position, value):  # <3> 
  26.         self._cards.insert(position, value) 

通過抽象基類collections.MutableSequence源碼:

可以發現,它有三個抽象方法__setitem__、__delitem__、insert,所以FrenchDeck2類必須實現它們。而對于其他非抽象方法比如append、extend、pop等,則可以直接繼承無需實現。

注意,Python只會在運行時實例化FrenchDeck2類時真正檢查抽象方法的實現,如果未實現會拋出TypeError異常,提示Can't instantiate abstract class之類的。

標準庫中的抽象基類

為了知道哪些抽象基類可以使用,我們可以看看標準庫。

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。

定義抽象基類

本小結可以跳過。不過了解抽象基類的定義有助于閱讀標準庫和其他包中的抽象基類源碼。

抽象基類的示例代碼如下:

  1. BEGIN TOMBOLA_ABC 
  2.  
  3. import abc 
  4.  
  5. class Tombola(abc.ABC):  # <1> 
  6.  
  7.     @abc.abstractmethod 
  8.     def load(self, iterable):  # <2> 
  9.         """Add items from an iterable.""" 
  10.  
  11.     @abc.abstractmethod 
  12.     def pick(self):  # <3> 
  13.         """Remove item at random, returning it. 
  14.         This method should raise `LookupError` when the instance is empty. 
  15.         ""
  16.  
  17.     def loaded(self):  # <4> 
  18.         """Return `True` if there's at least 1 item, `False` otherwise.""" 
  19.         return bool(self.inspect())  # <5> 
  20.  
  21.  
  22.     def inspect(self): 
  23.         """Return a sorted tuple with the items currently inside.""" 
  24.         items = [] 
  25.         while True:  # <6> 
  26.             try: 
  27.                 items.append(self.pick()) 
  28.             except LookupError: 
  29.                 break 
  30.         self.load(items)  # <7> 
  31.         return tuple(sorted(items)) 
  32.  
  33.  
  34. END TOMBOLA_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. class TomboList(list): 
  3.     ... 

白鵝類型和鴨子類型是Python的動態特性,它們的共同點是,只要長的像,Python就不會做強制檢查,鴨子類型是針對普通類的子類而言的,白鵝類型是針對抽象基類的虛擬子類而言的。

參考資料:

 

《流暢的Python》第11章 接口:從協議到抽象基類

 

責任編輯:武曉燕 來源: dongfanger
相關推薦

2024-12-25 08:00:44

Python抽象基類JSON

2009-07-28 17:38:02

ASP.NET多態抽象基類

2025-02-07 08:47:38

C#派生類接口

2009-12-21 16:24:24

WCF新到工廠

2009-08-03 18:12:31

C#抽象類

2023-12-08 14:50:45

Python枚舉類工具

2010-07-06 08:58:52

UML圖表達C++

2009-08-14 15:54:17

C#接口和抽象類

2009-08-04 17:42:23

DataSourceCASP.NET

2011-06-28 10:55:20

C#接口抽象類

2010-01-27 10:22:53

C++基類

2010-01-21 13:33:44

C++基類

2011-10-25 09:52:56

jQuery

2022-05-11 15:06:02

MySQL游標SQL

2015-09-08 11:06:46

設計編輯窗體

2009-07-22 18:08:00

ASP.NET基類

2009-07-14 13:49:29

原型

2009-06-29 15:15:00

抽象類Java

2009-12-02 14:55:46

PHP抽象類abstr

2012-02-29 09:32:01

Java
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产资源一区二区三区 | 成人在线视频免费播放 | 亚洲精品久久久一区二区三区 | 99re66在线观看精品热 | 国产视频1 | 久久成人在线视频 | 日韩精品av一区二区三区 | 亚洲欧美成人 | 日韩免费一区 | 亚洲午夜精品一区二区三区他趣 | 在线观看中文字幕av | 国产一区二区在线91 | 欧美精品一区二区三区在线播放 | 日韩中文在线视频 | 涩涩导航 | 毛片链接| 久久久成人精品 | 精品国产18久久久久久二百 | 国产91成人 | 亚洲视频国产 | 一区二区三区不卡视频 | 国产女人与拘做受视频 | 一级黄色毛片免费 | 日韩精品一区二区三区在线观看 | 中文字幕在线一区二区三区 | 国内自拍第一页 | 国产a区 | 国产999精品久久久久久绿帽 | 国产清纯白嫩初高生视频在线观看 | 天堂一区二区三区 | 婷婷激情综合 | 成人久久18免费网站图片 | 自拍 亚洲 欧美 老师 丝袜 | 精品1区 | 天天操天天射天天舔 | 一区二区三区在线免费观看视频 | 免费观看一级毛片 | 久久国产视频网 | 国产高清免费 | 国产欧美一区二区三区另类精品 | 国产精品视频999 |