继承
一、什么是继承
继承是一种关系,描述两个对象之间,什么是什么的关系
在程序中,继承描述的是类和类之间的关系
例如a继承了b,a就能直接使用b已经存在的方法和属性
a称之为子类,b称之为父类,也称之为基类
class A:
text = 'a'
def a(self):
print('from a')
pass
class B(A):
def b(self):
print('from b')
pass
b = B()
print(b.text)
b.a()
# 输出
# a
# from a
二、为什么要使用继承
继承的一方可以直接使用被继承一方已经有的东西
其目的是为了重用已经有的代码,提高重用性
三、如何使用继承
class A:
pass
class B(A):
pass
# B继承了A
四、抽象
将多个子类中相同的部分,进行抽取,形成一个新的类,这个过程也称之为抽象的过程
正确的使用继承:
先抽象再继承
继承一个已经存在的类,扩展或是修改原始的功能
# 抽取老师和学生中相同的部分形成person类
class Person:
def __init__(self, name, age, gender):
self.name = name
self.age = age
self.gender = gender
def say_hi(self):
print(f"name:{self.name},gender:{self.gender},age:{self.age}")
class Teacher(Person):
def teaching(self):
print("老师教学生写代码")
class Student(Person):
pass
五、属性的查找顺序
- 对象自己的
- 所在类的
- 父类的
- 父类的父类的
- ...
- object
六、派生
当一个子类中出现了与父类中不同的内容时,这个子类就称之为派生类
通常子类都会写一些新的代码,不可能和父类完全一样,即通常都是派生类
七、覆盖/重写(overrides)
当子类出现了与父类名称一样完全一致的属性或方法
# 实现一个能够限制元素类型的列表类
class MyList(list):
def __init__(self, element_type):
super().__init__() # 调用父类的方法初始化方法,来完成基本的初始化
self. element_type = element_type
def append(self, object):
'''
:param object: 是要存储的元素
:return: 没有
'''
if type(object) == self.element_type:
# 我们需要在这里访问父类的append函数来完成真正的存储操作
super(MyList, self).append(object)
else:
print("sorry sir, this element is not %s"%self.element_type)
# 创建指定存储类型的对象
m = MyList(int)
# 当你有需求, 是需要在创建对象时,干点什么事,那就该想到初始化方法
m.append(1)
print(m[0])
m.append("1212")
八、子类访问父类中的内容
# 方式一 python3和python2中都可以使用
super(当前类名称,self).你要调的父类的属性或方法
# 方式二 只能在python3中使用,也是最常用的方式
super().你要调的父类的属性或方法
# 方式三
类名称.你要调的父类的属性或方法
# 方式三与继承无关
强调:当你继承一个现有的类,并且覆盖了父类的init方法时,必须在初始化方法的第一行调用父类的初始化方法,并传入父类所需的参数
九、组合
也是一种关系,描述两个对象之间,什么有什么的关系
例如,学生有手机,游戏中角色拥有某些装备
将一个对象作为另一个对象的属性(即什么有什么)
十、组合的目的
为了重用现有代码
什么时候使用继承:分析两个类的关系,到底是不是:什么是什么的俄关系
什么时候使用组合:两个类之间没有太大的关系,完全不属于同类
组合相比继承,耦合度降低
十一、菱形继承
前提:python支持多继承
补充:新式类和经典类
新式类:任何显式或隐式的继承自object的类就称之为新式类,python3中全是新式类
经典类:不是object的子类,仅在python2出现
当出现了菱形继承时,新式类的属性查找顺序为先深度,如果遇到了共同父类时就广度
经典类的属性查找顺序为深度优先,没有广度查找