首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >怎么理解Python类中的super函数

怎么理解Python类中的super函数

作者头像
罗罗攀
发布2021-07-08 10:59:46
7900
发布2021-07-08 10:59:46
举报

前言

在Python类的继承中,经常能看到super函数的存在,那super函数主要的作用,以及如何理解和使用好这个函数?本次教程将详细讲解,希望大家看到最后,并按照代码实际操作下。

常见用途

我们举一个简单例子,我们父类是Human,有两个属性,分别是姓名和性别;然后定义一个子类Student。

class Human:

    def __init__(self, name, sex):
        self.name = name
        self.sex = sex
        print('parent')


class Student(Human):

    pass


stu_1 = Student('lisi', 'male')
print(stu_1.name)

parent
lisi

这里Student没有构造函数,所以会去父类中寻找构造函数。这时候我们需要在子类中加入构造函数,并需要name,sex,score三个属性,那我们如果直接写就应该是下面这种代码。

class Human:

    def __init__(self, name, sex):
        self.name = name
        self.sex = sex
        print('parent')


class Student(Human):

    def __init__(self, name, sex, score):
        self.name = name
        self.sex = sex
        self.score = score
        print('child')

stu_1 = Student('lisi', 'male', 97)
print(stu_1.score)

child
97

你会发现父类和子类都有相同的两行代码。

self.name = name
self.sex = sex

这显然和我们优雅的Python格格不入,所以super函数来了,我们直接看代码。

class Human:

    def __init__(self, name, sex):
        self.name = name
        self.sex = sex
        print('parent')


class Student(Human):

    def __init__(self, name, sex, score):
        super().__init__(name, sex)
        self.score = score
        print('child')

stu_1 = Student('lisi', 'male', 97)
print(stu_1.score)

parent
child
97

通过代码我们可以看出,super函数常常用于子类的构造函数中,用于调用父类(超类)的构造函数,并且不会显式引用基类。

之所以说不用显式引用基类,是因为通过调用父类方法也能实现功能。

class Human:

    def __init__(self, name, sex):
        self.name = name
        self.sex = sex
        print('parent')


class Student(Human):

    def __init__(self, name, sex, score):
        Human.__init__(self, name, sex)
        self.score = score
        print('child')

stu_1 = Student('lisi', 'male', 97)
print(stu_1.score)

parent
child
97

理解super函数

我将通过下面几个进阶的知识点让你理解super函数。

首先我们还是来总结下super函数的语法。

super(type[, object-or-type])

super(Student, self).__init__()  #python2写法
super().__init__()  #python3写法
不仅仅是用于构造函数

super函数虽常用于构造函数,但是父类的其他函数一样也是可以用super函数的。

class A:
    def add(self, x):
        y = x + 1
        print(y)


class B(A):
    def add(self, x):
        super().add(x)


b = B()
b.add(2)

# 3

之所以不常用,我认为是既然继承了父类,那子类就可以直接调用父类的方法,这样做只是多此一举。

class A:
    def add(self, x):
        y = x + 1
        print(y)


class B(A):
    pass


b = B()
b.add(2)

# 3
多重继承可能就不一样了

根据上面的案例,我们可以看出super函数是直接调用基类的构造函数,但是多重继承不一样,他是调用继承顺序的下一个类,而不是父类。

class Base:
    def __init__(self):
        print("enter Base")
        print("leave Base")

class A(Base):
    def __init__(self):
        print("enter A")
        super().__init__()
        print("leave A")

class B(Base):
    def __init__(self):
        print("enter B")
        super().__init__()
        print("leave B")

class C(A, B):
    def __init__(self):
        print("enter C")
        super().__init__()
        print("leave C")

C()

enter C
enter A
enter B
enter Base
leave Base
leave B
leave A
leave C

继承关系为C—A—B—Base,所以程序会先去A,再到B,最后到Base。

super函数和直接调用父类方法的区别

在单继承时,我们看到super和直接调用父类方法得到的结果是一样的,只是不会显式引用基类。但多重继承就不要了,我把上面的代码进行了修改,我相信你能看懂区别。

class Base:
    def __init__(self):
        print("enter Base")
        print("leave Base")

class A(Base):
    def __init__(self):
        print("enter A")
        Base.__init__(self)
        print("leave A")

class B(Base):
    def __init__(self):
        print("enter B")
        Base.__init__(self)
        print("leave B")

class C(A, B):
    def __init__(self):
        print("enter C")
        A.__init__(self)
        B.__init__(self)
        print("leave C")

C()

enter C
enter A
enter Base
leave Base
leave A
enter B
enter Base
leave Base
leave B
leave C

最后,由于本人时间和能力有限,如有错误,请批评指正,我们下期再见~

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 常见用途
  • 理解super函数
    • 不仅仅是用于构造函数
      • 多重继承可能就不一样了
        • super函数和直接调用父类方法的区别
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档