一摞python风格的纸牌(fluent python阅读)
? ? ? ? ? ? ? ? ? ? ? ? ? 這一節會展示如何實現__getitem__和__len__這兩個特殊的方法
首先看一個利用python構建的紙牌類,這個例子短小卻包含了python一些非常重要的特征:
import collectionsCard = collections.namedtuple('Card', ['rank', 'suit'])class FrenchDesk(object):ranks = [str(i) for i in range(2, 11)] + list('JQKA')suits = 'spades diamonds clubs hearts'.split()def __init__(self):self._cards = [Card(rank, suit) for suit in self.suitsfor rank in self.ranks]def __len__(self):return len(self._cards)def __getitem__(self, position):return self._cards[position]desk = FrenchDesk() print(len(desk)) print(desk[-1])Return: 52 Card(rank='A', suit='hearts')collections.namedtuple
Python中存儲系列數據,比較常見的數據類型有list,除此之外,還有tuple數據類型。相比與list,tuple中的元素不可修改,在映射中可以當鍵使用。tuple元組的item只能通過index訪問,collections模塊的namedtuple子類不僅可以使用item的index訪問item,還可以通過item的name進行訪問,可以用來增強代碼的可讀性。可以將namedtuple理解為c中的struct結構,其首先將各個item命名,然后對每個item賦予數據。
len(desk) 與desk[index]
由于實現了__len__(self)方法,所以可以直接使用len(desk)獲取長度,作為你的類的用戶,不必去記住標準操作的各式名稱(“怎么得到元素的總數?是 .size() 還是 .length() 還是別的什么?”)。
同樣,?__getitem__ 方法把 [] 操作交給了 self._cards 列表,所以 deck 類還自動支持切片(slicing)操作。另外,僅僅實現了 __getitem__ 方法,這一摞牌就變成可迭代的了。
?
總結
雖然 FrenchDeck 繼承了 object 類, 但功能卻不是繼承而來的。通過數據模型和一些合成來實現這些功能。通過實現 __len__ 和 __getitem__ 這兩個特殊方法,FrenchDeck 就跟一個 Python 自有的序列數據類型一樣,可以體現出 Python 的核心語言特性(例如迭代和切片)。同時這個類還可以用于標準庫中諸如 random.choice、reversed 和 sorted 這些函數。另外,對合成的運用使得 __len__ 和 __getitem__ 的具體實現可以代理給 self._cards 這個 Python 列表(即 list 對象)。?
總結
以上是生活随笔為你收集整理的一摞python风格的纸牌(fluent python阅读)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ul和ol的区别以及经验总结
- 下一篇: python django 动态网页_D