python中一个类有多个父类的时候,默认使用第一个父类的同名属性和方法;
如果子类先调用父类的attribut和method, 父类属性会覆盖子类属性,所以在调用父类时先调用子类的初始化self.init() 注意括号中为空,具体解释看后面代码示例;
python中子类继承父类:
子类对象调用父类attribut、method的三种方式:
方法 1
父类名.init(self, 父类同类参数)
父类名.method(父类同类参数)方法2
super(子类名,self).init(父类同类参数)
super(子类名,self).method(父类同类参数)方法3
super().init(self,父类同类参数)
super().method(父类同类参数)方法2和方法3省略括号里内容的区别**,一般用方法1 和 方法3**
**
父类名.init(self, 参数) | 父类名; ()中一定有self +参数 |
---|---|
super(子类名,self).init(参数) | 子类名; ()中参数一定没self |
# python中重写父类方法,不用函数名相同,但是参数要相同
# python中有重载的思想但是没有重载行为,python是动态语言不需要重载
class testA(object):
def __init__(self,name):
self.name=name
self.age=100
def sMethod(self, sMethodAttribut):
sMethodAttribut="fu_sMethodAttribut"
print(sMethodAttribut)
class testB(testA):
def __init__(self):
self.age=age
""" 调用子类自己的初始化。
子类调用父类的属性和方法时,父类属性会覆盖子类属性
所以调用父类属性前,先调用自己子类的初始化(这里讲的始终是属性,不涉及方法)"""
def zMethod(self,name):
print("zi method, self : ",self)
self.__init__() #()里是空的,与子类初始化init函数中参数一致
"""重写父类方法,子父类方法名可以不同,但是参数要一致;
python不会自动调用父类的constructeur,需要手动调用,
所以调用父类方法时,为保证调用到的也是父类的属性
必须在调用父类方法前调用父类的初始化,也就是调用父类的__init__(self,参数)
而,如何调用父类的init函数,调用方式同上文的3种方式。"""
"""子类重写父类方法 1"""
def sup1(self,name):
print("fu method, self : ",self)
testA.__init__(self,name) #这里是调用父类的init函数,init中传递的参数要和父类保持一致, ()里一定有self
testA.sMethod(self,name) # 这里调用的是父类的method,所以()里参数和父类保持一致,一定有self
"""子类重写父类方法 2"""
def sup2(self,name):
print("fu method, self : ",self)
super(testB,self).__init__(name) # 这里是调用父类的init函数,init中传递的参数要和父类保持一致(后面的括号中一定没有self)
super(testB,self).sMethod(name) # ()里参数和父类保持一致
"""子类重写父类方法 3"""
def sup3(self,name):
print("fu method, self : ",self)
super().__init__(name) #这里是调用父类的init函数,init里参数和父类保持一致,没有self
super().sMethod(name) #()里参数和父类保持一致
# 创建实例/对象
b=testB()
# 对象调用子类方法
b.zMethod("immuable") # zi method, self : <__main__.testB object at 0x00000195CC3D9550>
#对象调用父类方法
b.sMethod("we") # fu_sMethodAttribut
# 对象重写父类的3种方法
print("\n-------------------------super 1--------------------")
b.sup1("super 1")
print("\n-------------------------super 2--------------------")
b.sup2("super 2")
print("\n-------------------------super 3----------------------")
b.sup3("super 3")
输出
# 对象调用子类方法
zi method, self : <__main__.testB object at 0x00000195CC99EB70>
#对象调用父类方法
fu method
# 对象重写父类的3种方法
-------------------------super 1--------------------
fu method, self : <__main__.testB object at 0x00000195CC99EB70>
super 1
-------------------------super 2--------------------
fu method, self : <__main__.testB object at 0x00000195CC99EB70>
super 2
-------------------------super 3----------------------
fu method, self : <__main__.testB object at 0x00000195CC99EB70>
super 3
python类中初始化__init__()方法是可选的, 如果子类没有新的构造参数,或 新的初始化逻辑,则初始化__init__()方法没有必要被创建,因为这里python会找到父类的初始化方法并执行。
"""子类继承父类的属性时:子类需要先调用父类的init函数,因为子类的init函数不会自动继承父类中init函数中的属性。最后通过实例化对象调用父类中的属性报错"""
class A(object):
def __init__(self,name):
self.name=name
class B(A):
def __init__(self,name,age):
self.age=age
b=B("immuable",18)
b.age
# b.name #报错, 因为父类的属性并没有被调用
输出
18
如下是想要调用父类属性name的正确写法
class A(object):
def __init__(self,name):
self.name=name
def fuMethod(self):
print("fu method")
class B(A):
def __init__(self,name,age):
super().__init__(name) # 需要调用父类中的init函数
self.age=age
def fuMethod(self):
print("zi method")
b=B("immuable",18)
print(b.age, b.name)
# 子类继承父类时 :通过实例化对象调用父类方法,但这种情况下,子类的同名方法就会覆盖 父类同名方法(奖励讲的是方法,不涉及属性)。属性见例子1或例子3
b.fuMethod() #
输出
在这里插入代码片
class A(object):
def __init__(self,name):
self.name=" fu name"
def fuMethod(self):
print("fu method")
class B(A):
def __init__(self,name,age):
super().__init__(name)
self.age=age
#子类中与父类同名的方法: 子类方法覆盖父类方法
#子类继承父类属性name:这里先调用父类属性,那么父类属性name覆盖子类属性name
# 但一般都是self.name=name这种正常的写法,通过self.name=‘fu name’可以看到参数的传递
def fuMethod(self,name,age):
self.name="zi name"
print("zi method")
b=B("immuable",18)
print(b.age, b.name)
b.fuMethod("--",18)
输出
fu name 18
zi method
例子3的正常写法及其输出
class A(object):
def __init__(self,name):
self.name=name # 改成正常形式
def fuMethod(self):
print("fu method")
class B(A):
def __init__(self,name,age):
self.name="zi name" #这里还是写成赋值形式,便于查看参数传递
self.age=age
super().__init__(name)
def fuMethod(self,name,age):
self.__init__(name,age)
self.name="zzi name" #这里还是写成赋值形式,便于查看参数传递
print("zi method")
b=B("immuable",18)
print(b.name,b.age)
b.fuMethod("--",18)
immuable 18
zi method
参考:http://t.csdn.cn/syJds