Python super()函数

Python 中子类会继承父类所有的类属性和类方法。严格来说,类的构造方法其实就是实例方法,因此毫无疑问,父类的构造方法,子类同样会继承。
如果子类继承的多个父类中包含同名的类实例方法,则子类对象在调用该方法时,会优先选择排在最前面的父类中的实例方法。

举个栗子:

class Dad:
    def __init__(self,name):
        self.name = name
    def say(self):
        print("我是你爸爸,名字叫:",self.name)
class Son:
    def __init__(self,food):
        self.food = food
    def display(self):
        print("我是你儿子,我喜欢吃",self.food)

#Dad中的 name 属性和 say() 会遮蔽 Son 类中的food
class Person(Dad, Son):
    pass
per = Dad("zhangsan")
per.say()
per.display()

### 输出结果:
我是你爸爸,名字叫: zhangsan
Traceback (most recent call last):
  File "tests.py", line 107, in <module>
    per.display()
  File "tests.py", line 98, in display
    print("我是你儿子,我喜欢吃", self.food)
AttributeError: 'Person' object has no attribute 'food'

上面程序中,Person 类同时继承 Dad 和 Son,其中 Dad 在前。这意味着,在创建 per 对象时,其将会调用从 Dad 继承来的构造函数。因此我们看到,上面程序在创建 per 对象的同时,还要给 name 属性进行赋值。
但由于 Dad 类的构造方法“遮蔽”Son 类的构造方法,使得在创建 per 对象时,Son 类的构造方法未得到执行,不存在food 属性的值, 所以程序出错。

假如此时将继承顺序互换

class Person(Son, Dad):

那么一样Son类的构造方法“遮蔽”Dad 类的构造方法,Dad 类的构造方法未得到执行,不存在name属性的值, 所以程序还是会出错。

针对这种情况,正确的做法是定义 Person 类自己的构造方法(等同于重写第一个直接父类的构造方法)。但需要注意,如果在子类中定义构造方法,则必须在该方法中调用父类的构造方法。

在子类中的构造方法中,调用父类构造方法的方式有 2 种,分别是:

  1. 类可以看做一个独立空间,在类的外部调用其中的实例方法,可以向调用普通函数那样,只不过需要额外备注类名(此方式又称为未绑定方法);
  2. 使用 super() 函数。但如果涉及多继承,该函数只能调用第一个直接父类的构造方法。

Python 2.x 中,super() 函数的使用语法格式如下:

super(Class, obj).__init__(self,...)

在 Python 3.x 中,super() 函数有一种更简单的语法格式,推荐使用这种格式:

super().__init__(self,...)
class People:
    def __init__(self,name):
        self.name = name
    def say(self):
        print("我是人,名字为:",self.name)
class Animal:
    def __init__(self,food):
        self.food = food
    def display(self):
        print("我是动物,我吃",self.food)
class Person(People, Animal):
    #自定义构造方法
    def __init__(self,name,food):
        #调用 People 类的构造方法
        super().__init__(name)
        #super(Person,self).__init__(name) #执行效果和上一行相同
        #People.__init__(self,name)#使用未绑定方法调用 People 类构造方法
        #调用其它父类的构造方法,需手动给 self 传值
        Animal.__init__(self,food)    
per = Person("zhangsan","熟食")
per.say()
per.display()
### 运行结果
我是人,名字为: zhangsan
我是动物,我吃 熟食
posted @ 2022-05-05 00:10  大切切  阅读(107)  评论(0编辑  收藏  举报