python中的super怎么用?

时间:2022-11-06 00:22:27

面向对象有这个强大特点和作用, 著名的三大特点:封装, 继承, 多态

这篇博客写的是super()的简单理解和使用

今天在读restframework的源码的时候, 发现源码中使用了super, 依以此为入口, 重写了django的as_view()

在代码执行的过程中既执行了自己的as_view()有执行了django的as_view()

super()能做什么?

之前的理解就是, 使用super()可以在子类中调用父类的方法或属性, 可能你会说, 子类本来就可以调用父类中所有非私有的属性或方法,

而我现在说的是, 当子类中实现了某个方法, 父类中也有这个方法, 当你调用这个方法时, 既想执行子类的又想执行父类的, 在这种情况下

就可以使用super()

super()怎么用的

class A(object):
def __init__(self, name, age):
self.name = name
self.age = age
print("class A") class B(A):
def __init__(self, name, age):
super().__init__(name, age)
print("class B") class C(B):
def __init__(self, name, age):
# 方式一
super().__init__(name, age)
# 相当于 super(C, self).__init__(name, age)
# 默认是从当前类的父类开始往上找__init__() # 方式二
# 指定从当前类的某个父类中开始往上找
super(B, self).__init__(name, age)
print("class C") c = C("sath", 37)

定义了三个类, 继承关系为 C 继承 B 继承 A

定C类进行实例化的时候, 要先执行__init__()[__new__先不说],

C类中有__init__()方法, 所以就会执行C类中的__init__()

但是在C类自己的__init__()中却使用了super(). 并且是   点.__init__()

这就表示要去执行父类的__init()方法

super()有两中使用方法

方法一

super().__init__()

默认执行当前类的父类中的方法, 并且将当前类的实例对象self传了进去

其实就相当于super(C, self).__init__()

父类中的__init__(self), 中的self就是super()传进去的self,

方法二

super(B, self)>__init__()

其实和方法一一样, 就是指定的父类不一样了, 你可以指定执行那个父类的__init__()

前提是必须是当前这个类的父类

这样执行的执行的话, 参数就会灵活很多

restframework中对super()的应用实例

先来看一下源码

@classmethod
def as_view(cls, **initkwargs):
if isinstance(getattr(cls, 'queryset', None), models.query.QuerySet):
def force_evaluation():
raise RuntimeError(
'Do not evaluate the `.queryset` attribute directly, '
'as the result will be cached and reused between requests. '
'Use `.all()` or call `.get_queryset()` instead.'
)
cls.queryset._fetch_all = force_evaluation view = super(APIView, cls).as_view(**initkwargs)
super().__init__()
view.cls = cls
view.initkwargs = initkwargs return csrf_exempt(view)

现在就来看这行代码 view = super(APIView, cls).as_view(**initkwargs)

写了一个CBV, 视图继承了APIView

当请求进来之后, 进行路由匹配然后调用视图函数, CBV会执行as_view()

我是的路由是这样式的

url(r'^auth/', views.AuthView.as_view())

AuthView就是视图类, 继承了APIView, 当调用as_view()时, 先去自己的类中找as_view, 发现没有, 然后去父类中找

上面的as_view()代码就是APIViwe类中的as_view(), APIView其实也继承了View

发现as_view()是一个类方法, 怪不得在路由中使用类去调用呢, 这个样就能得到视图类的类空间了

因为在View中需要用视图类, 然后对视图类进行了实例化, 通过这个实例在加上反射获取和用户请求方式同名的函数并执行

上面就是super()在restframework中的应用