目录
在Python程序中,类的继承是指新类从已有类中取出已有的特性,例如:属性、方法等。类的派生是指从已有的类产生新类的过程,这个已有的类被称为基类或者父类。而新类则被称为派生类或者子类。字类不但可以继承使用父类中的数据成员(类变量、对象变量)和成员函数(方法),而且可以增加新的成员。
在Python中类的继承是多继承
1.1定义字类
继承语法:
在Python中,类的继承是多继承:一个子类可以继承多个父类
class ClassName(ParentClass1, ParentClass2, ParentClass3...):
statement
class Delivery:
pass
"""
== class Delivery(object):
pass
上边省略了它的继承,为什么省略?
因为object是我们所有类的基类, 无论哪个类它其实都是继承自object
"""
1.1.1子类的使用
上代码(单继承):
#定义父类
class Parent:
count = 0 #类变量
#带参构造方法
def __init__(self, name, age):
self.age = age #对象变量
self.name = name #对象变量
#定义方法
def print_info(self):
print(f"{self.name}-{self.age}")
#定义子类继承父类
class Son(Parent):
pass
#实例化对象
son = Son("son", 10) #son对象有对象变量: name, age
print(son.name)
print(son.age)
son.print_info()
print(son.count)
# 实例化Son类的时候: __new__ -> __init__
# 一个是创建并返回一个新的对象 __new__(不参与)
# 初始化返回的这个对象 (可以参与的) __init__
# Son继承自Parent,所以它在实例化的时候:
# 先去调用__new__(Son未定义) -> Parent找(也没有定义)->object中找,找到了
# 去调用__init__(Son为定义) -> Parent找(定义了),找到了
执行结果:
多继承:
class A: #定义父类A,父类A继承(object)(object 可省略)
def print_info(self):
print("This is A")
class B:
def print_info(self):
print("This is B")
def say_something(self):
print("My name is B")
class D:
def say_something(self):
print("My Name is D")
#子类C继承父类A,父类B和父类C
class C(A, B, D):
pass
# C和E有一个相同Parent A, A 是有两个孩子 C和E, C和E是兄弟类
class E(A):
pass
c = C()
# 为什么在执行c.print_info(),调用的是A中print_info方法,而不是B中的?
c.print_info()
c.say_something()
#Python是可以支持多继承
#继承顺序是什么? Python继承的顺序是依靠一个叫做C3线性算法
print(C.__mro__) #mro: method resolution order: 查找顺序
#如果我们要改变它的继承顺序,如何去改变?: 使用super()关键字
执行结果:
1.1.3 super()关键字
super的使用格式:
super([type[, object-or-type]])
[type[, object-or-type]]
中括号:代表的是这些参数可以传,也可以不传
可以什么都不传
可以只传递一个type
可以传递: type和object-or-type
super的作用:返回一个代理对象,它会将方法调用委托给 type 的父类或兄弟类。
type: 类
object-or-type: 可以是一个对象也可以是一个类
object-or-type 确定用于搜索的 method resolution order,
搜索会从 type 之后的类开始。
如果 object-or-type 的 __mro__ 为 D -> B -> C -> A -> object
并且 type 的值为 B,
则 super() 将会搜索 C -> A -> object。
限定条件:如果省略第二个参数,则返回的超类对象是未绑定的。
如果第二个参数为一个对象,则 isinstance(obj, type) 必须为真值。
如果第二个参数为一个类型,则 issubclass(type2, type) 必须为真值。
上代码:
class A(object):
def print_info(self):
print("This is A")
class B:
def print_info(self):
print("This is B")
def say_something(self):
print("My name is B")
class D:
def say_something(self):
print("My Name is D")
class F(A, B, D):
def print_info(self):
super(A, F).print_info(self)
def say_something(self):
super(B, F).say_something(self)
pass
f = F()
f.print_info()
f.say_something()
print(F.__mro__)
执行结果:
小总结:
继承:
1.Python支持单继承和多继承
class ClassName(Parent1...):
2.object是所有类的父类(基类)
class ClassName:
== class ClassName(object):
3.super:
就可以将我们方法的调用委托给父类或者兄弟类
因为它可以改变我们继承的顺序
4.__mro__: 继承的搜索顺序:利用了C3线性算法
1.2 _和__的使用
- 在类中想要实现私有和不被继承的东西:_和__(python中没有私有这个概念)
- _命名的方法或变量, 是在python中约定为私有变量或私有方法的
因为只是一个约定,实际上在继承时候是可以继承到的
- __命名的方法或变量,在python解释器会对名称做一个转换:
转换的格式:_ClassName+methodName
__test_func() -> _Person__test_func()
__test_func() -> _Man__test_func()
作用是什么?
他的作用: 避免我们子类和父类中名称的冲突
代码如下:
class Person:
def __init__(self, name, age, gender):
self.name = name
self.age = age
self._gender = gender
def print_info(self):
print(f"{self.name}-{self.age}-{self._gender}")
def __test_func(self):
print("This is testing double_underline")
class Man(Person):
def __test_func(self):
print("This is Man Testing double_underline")
pass
man = Man("zhangsan", 18, "男")
print(man.name)
print(man.age)
print(man._gender) # access to a protected member _gender
# Python约定的一个私有变量: 使用_xxx来命名
# 这个意义在于:如果外部的人去使用这个对象的时候:
# 使用对象.可以访问name和age,但是他是不知道_gender的
# _gender是受保护
# 访问一个受保护的成员_gender
man.print_info()
# man.__test_func() # 'Man' object has no attribute '__test_func'
# print(dir(man))
#_Person__test_func
man._Person__test_func()
man._Man__test_func()
执行结果:
方法的重写:
也叫方法的覆盖,如果从父类继承的方法不能满足子类的需求,可以对其进行重写,这个叫做,方法的覆盖,也叫做方法的冲写。