为什么需要super
在python没有引入super之前, 如果需要在子类中引用父类的方法, 一般写法如下:
1
2
3
4
5
6
7
8
9
|
class Father: def whoami( self ): print ( "I am father" ) class Child: def whoami( self ): print ( "I am child" ) Father.whoami( self ) |
这样看好像没什么问题, 就算没有super也能正常调用父类的方法, 但是如果有一天Father类需要修改类名为Father1, 那么子类Child中也必须跟着修改.
想象下如果一个类有很多个子类, 这样一来我们就需要修改每个子类中引用父类的语句
怎么使用super
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
Help on class super in module builtins: class super(object) | super() -> same as super(__class__, <first argument>) | super(type) -> unbound super object | super(type, obj) -> bound super object; requires isinstance(obj, type) | super(type, type2) -> bound super object; requires issubclass(type2, type) | Typical use to call a cooperative superclass method: | class C(B): | def meth(self, arg): | super().meth(arg) | This works for class methods too: | class C(B): | @classmethod | def cmeth(cls, arg): | super().cmeth(arg) |
我们来看看super的帮助文档, 首先super是一个类, 它的调用方法有如下几种:
1.super()
2.super(type)
3.super(type, obj)
4.super(type, type2)
我们推荐用第一种方法来使用super, 因为它并不需要显式传递任何参数.但是要注意一点, super只能在新式类中使用.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
class A: def __init__( self ): print ( "this is A" ) class B(A): def __init__( self ): super ().__init__() print ( "this is B" ) b = B() """ this is A this is B """ |
看起来super就像直接调用了B的父类A的__init__, 其实并不是这样的, 我们看看super在多继承下的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
|
class A: def __init__( self ): print ( "this is A" ) print ( "leave A" ) class B(A): def __init__( self ): print ( "this is B" ) super ().__init__() print ( "leave B" ) class C(A): def __init__( self ): print ( "this is C" ) super ().__init__() print ( "leave C" ) class D(B, C): def __init__( self ): print ( "this is D" ) super ().__init__() print ( "leave D" ) d = D() """ this is D this is B this is C this is A leave A leave C leave B leave D """ print (D.__mro__) """ (<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>) """ |
这里可以看到, 如果super只是简单调用父类的方法的话, 那么调用了B的__init__ 方法之后, 应该调用A的__init__ 才对, 但是调用的却是C的__init__ 方法
这是因为super调用的是当前类在MRO列表中的后一个类, 而并不是简单地调用当前类的父类
python并没有强制性限制我们使用super调用父类, 我们还是可以使用原始的方法来调用父类中的方法, 但是需要注意的是调用父类的方法要统一, 即全用super或全不用super, 而用super 的话, 调用的方式也最好是统一的
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://www.cnblogs.com/dears/p/9138268.html