记住以下几点:
直接子类化内置类型(如
dict
,list
或str
)容易出错,因为内置类型的方法通常会忽略用户覆盖的方法,不要子类化内置类型,用户自定义的类应该继承collections
模块.def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) # 错误案例 class AnswerDict(dict): def __getitem__(self, item): # 错误案例 return 42 import collections class DoppelDict2(collections.UserDict): # 正确案例 def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict2(collections.UserDict): # 正确案例 def __getitem__(self, item): return 42
多重继承有关的另一个问题就是:如果同级别的超类定义了同名属性.
Python
如何确定使用哪个?class DoppelDict(dict): def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict(dict): def __getitem__(self, item): return 42 import collections class DoppelDict2(collections.UserDict): def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict2(collections.UserDict): def __getitem__(self, item): return 42 class A: def ping(self): print('Ping:', self) class B(A): def pong(self): print('pong:', self) class C(A): def pong(self): print('PONG:', self) class D(B, C): def ping(self): super().ping() print('post-ping:', self) def pingpong(self): self.ping() super().ping() self.pong() super().pong() C.pong(self) if __name__ == '__main__': d = D() print(d.pong()) # 输出来源于B print(C.pong(d)) #输出来源于C 超类的方法都可以直接调用,此时要把实例作为显示参数传入.
python
能区别调用的是哪个方法,通过方法解析顺序
>>> D.mro()
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
若想把方法调用委托给超类,推荐的方式是使用内置的super()函数.- 以下是对于
d.pingpong()
方法的解读 >>> self.ping()
-
Ping: <__main__.D object at 0x000002213877F2B0>
post-ping: <__main__.D object at 0x000002213877F2B0>
第一个调用的是self.ping()
,运行的是是D类的ping,方法. - 第二个调用的的是
super().ping()
,跳过D类的ping
方法,找到A类的ping
方法.Ping: <__main__.D object at 0x000002213877F2B0>
- 第三个调用的是
self.pong()
方法,根据__mro__
,找到B类实现的pong
方法.pong: <__main__.D object at 0x000002213877F2B0>
- 第四个调用时
super().pong()
,也是根据__mro__
,找到B类实现的pong
方法.pong: <__main__.D object at 0x000002213877F2B0>
- 以下是对于
- 第五个调用的是
C.pong(self)
,忽略了__mro__
,找到的是C类实现的pong
方法.PONG: <__main__.D object at 0x000002213877F2B0>