浅析python继承与多重继承
记住以下几点:
直接子类化内置类型(如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>