书中 9-10 章用实现一个Vector类
来了解一个类的运行和实现过程,作者实现的细节非常值得一看,这里就不搬运了,讲一下序列中切片的原理。
- 鸭子类型
- 切片原理
鸭子类型
在程序设计中,鸭子类型(英语:duck typing)是动态类型的一种风格。在这种风格中,一个对象有效的语义,不是由继承自特定的类或实现特定的接口,而是由”当前方法和属性的集合”决定。
通俗来讲,“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”。很像一句老话,“视其所以,观其所由,察其所安”,当然含义完全不一样。具体检查从想要它做什么行为入手,设计也从这个思想去做。
Python中创建任意完整的序列如Vector
都无需使用继承,只需要实现符合序列协议的方法。这里呢,协议其实就是非正式的接口,定义在文档里,换言之,想要实现切片,把 __len__
和 __getitem__
两个方法实现就行,这两个就是接口。
切片原理
一个简单的序列(没有切片):
class FrenchDeck:
# ...
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]
实现切片之前,观察切片有什么行为。
对Vector
实现切片:
def __getitem__(self,index):
cls=type(self)
if isinstance(index,slice):
return cls(self._components[index])
elif isinstance(index,numbers.Integral):
return self._components[index]
else :
msg='{cls.__name__} indices must be integers'
raise TypeError(msg.format(cls=cls))