文章目录
- 一、前言
- 二、什么是super
- 三、super的常用使用场景
一、前言
最近有粉丝向我咨询super
相关的问题,说网上搜索到的教程不够通俗易懂,看了之后还是不太理解。所以在这里基于我自己的理解来讲解一下super
。
二、什么是super
1.super
也是一个类,是的。他不是一个方法也不是一个内置的关键字。
class A:
pass
print(type(super(A)))
输出结果
<class 'super'>
直接通过查看super
的源码也可以看出它是一个类
另外,网上很多文章讲解super
就是用来调用父类方法的,这也是一个错误的观点!
假如我们有下面这样一个例子:
class A:
def __init__(self):
print("A")
class B(A):
def __init__(self):
print("B")
super().__init__()
class C(A):
def __init__(self):
print("C")
super().__init__()
class D(B, C):
def __init__(self):
print("D")
super().__init__()
D()
如果按照“super
就是用来调用父类的方法的”这样的理解来看,那上述代码的执行的过程应该为:
print("D")
—【调用super会先后执行B和C】—先执行B:print("B")
—【调用super执行A】—print("A")
—【调用super会先后执行B和C】—后执行C:print("C")
—【调用super执行A】—print("A")
执行结果理论应该为:D B A C A
但实际情况却不是这样的!
实际执行结果
D
B
C
A
所以说“super
就是用来调用父类的方法的”这个说法是错误的!
实际上super
的调用是遵循Python的【MRO(方法解析顺序)】来执行的,在Python3中,MRO是基于C3算法来实现的。
关于MRO和C3算法的讲解可以参考这篇文章http://kaiyuan.me/2016/04/27/C3_linearization/
三、super的常用使用场景
1.假如我们继承的多个父类有同名的方法,可以使用super
来指定使用哪个父类的方法
class A:
def test(self):
print('A')
class B:
def test(self):
print('B')
class C(A, B):
def __init__(self):
super().test() # 调用A类中的test方法
super(C, self).test() # 调用A类中的test方法
super(A, self).test() # 调用B类中的test方法
C()
输出结果
A
A
B
2.当我们在子类中使用父类的一个方法并且想对其做一定扩展又不想完全重写,那么使用super()就可以实现方法的增量修改:
举一个例子,如果我们想把list中的append
的方法改为中文添加
应该怎么做呢?
并且python中list调用append
方法是没有返回值的,我们想在添加元素操作成功后加一个返回值返回成功
又该如何操作呢?
首先看通过调用原来的list
的执行结果
a=list()
res=a.append(1)
print(res)
输出结果
None
可以看到调用append
方法后返回的值为None
现在我们通过super
进行重写,让其具有返回值并可以直接通过中文来调用append
:
class 列表(list):
def 添加(self, *args, **kwargs):
super().append(*args, **kwargs)
return "添加成功"
x = 列表()
res = x.添加(1)
print(res)
print(x)
输出结果
添加成功
[1]
super
实际上还是很常用的,比如在restfremework中,需要重写其响应结果的Response信息的时候,除了通过django的中间件实现,也可以使用super
重写其dispatch
来实现。