假设存在 Person 与 Student 两个类:
class Person(object):
def __init__(self):
self.name = "Tom"
def getName(self):
return self.name
class Student(Person):
def __init__(self):
self.age = 12
def getAge(self):
return self.age
if __name__ == "__main__":
stu = Student()
print stu.getName()
运行以上代码会报以下错误
Traceback (most recent call last):
File "test.py", line 18, in <module>
print stu.getName()
File "test.py", line 6, in getName
return self.name
AttributeError: Student instance has no attribute 'name'
报错原因:其他语言(java、php)有所不同,尽管 Student 类继承了 Person 类,但是并没有调用父类的 __init__()
方法,所以在此时子类是不可调用父类的方法的。
方法一:调用未绑定的父类的构造方法
class Person(object):
def __init__(self):
self.name = "Tom"
def getName(self):
return self.name
class Student(Person):
def __init__(self):
Person.__init__(self)
self.age = 12
def getAge(self):
return self.age
if __name__ == "__main__":
stu = Student()
print stu.getName()
在调用一个实例的方法时,该方法的 self 参数会被自动绑定到实例上(称为绑定方法)。但如果直接调用父类的方法(比如 Person.__init__()
),那么就没有实例会被绑定,这样就可以自由的提供需要的 self 参数,这种方法就称为未绑定方法。
通过将当前的实例作为 self
参数提供给未绑定方法, Student
类就能使用父类构造方法的所有实现,从而 name
变量被设置。
方法二:使用 super 函数
class Person(object):
def __init__(self):
self.name = "Tom"
def getName(self):
return self.name
class Student(Person):
def __init__(self):
super(Student, self).__init__()
self.age = 12
def getAge(self):
return self.age
if __name__ == "__main__":
stu = Student()
print stu.getName()
super函数会返回一个super对象,这个对象负责进行方法解析,解析过程其会自动查找所有的父类以及父类的父类。
super函数比在子类中直接调用未绑定方法更直观,但是其最大的优点是如果子类继承了多个父类,它只需要使用一次super函数就可以。