一、继承
面向对象的语言:c++,c#,java
面对对象的三大特性(继承、多态、封装) 其他语言都有
二、继承和抽象
实例---> 分类 ---> 汇总 逐渐抽象
编程实现:先抽象,后继承
继承的好处:继承能提高代码的重用性,还能规范代码
继承分为子类和父类
父类/超类/基类
子类/派生类
class Animal: #父类
def __init__(self,name,kind):
self.name =name
self.kind =kind
defsleep(self):
print('%s在睡觉'%self.name)
class Cat(Animal): #子类
defcatch(self):
print('%s抓老鼠'%self.name)
cat = Cat('小猫','咖啡猫')
print(cat.name)
cat.sleep()
cat.catch()
------------------小猫
小猫在睡觉
小猫抓老鼠
------------------
继承的语法:
class A:pass
class B(A):pass
class A: #父类
...
class B(A): #B 继承 A
...
三、单继承
继承与重用
重用:子类可以使用父类中的所有属性和方法,父类中所有的属性和方法都可以被子类使用了
子类可以使用父类中的名字(变量和方法)
class Animal: #父类
def __init__(self,name,kind):
self.name =name
self.kind =kind
defsleep(self):
print('%s在睡觉'%self.name)
class Cat(Animal): #子类
defcatch(self):
print('%s抓老鼠'%self.name)
classDog(Animal):
defwalk_the_dog(self):
print('溜%s' %self.name)
cat = Cat('小猫','咖啡猫')
dog = Dog('小狗','导盲犬')
cat.sleep()
dog.sleep()
-----------------小猫在睡觉
小狗在睡觉
-----------------
派生
子类在父类的基础上又新创新了自己需要的方法和属性
① 父类有的子类没有--- 子类对象直接调用,就会直接执行父类的方法
② 父类有的子类也有--- 子类对象调用 直接执行子类中的方法
--- 在子类中使用父类的名字:父类名、super()
classAnimal:
def __init__(self,name,kind):
self.name =name
self.kind =kind
defdrink(self):
print('%s在喝水'%self.name)
class Cat(Animal): #派生类
def __init__(self,name,kind,eye_color):
self.name =name
self.kind =kind
self.eye_color = eye_color #派生属性
def catch(self): #派生方法
print('抓老鼠')
当子类当中有被调用的方法的时候,子类的对象会直接选择子类中的方法或者变量,父类中的方法不会被执行
方法一:子类已存在同名的方法,调用父类的方法
classAnimal:
def __init__(self,name,kind):
self.name =name
self.kind =kind
defdrink(self):
print('%s在喝水'%self.name)
class Cat(Animal): #派生类
def __init__(self,name,kind,eye_color):
Animal.__init__(self,name,kind) #调用父类的方法
self.eye_color = eye_color #派生属性
def catch(self): #派生方法
print('抓老鼠')
方法二:子类已存在同名的方法,调用父类的方法
classAnimal:
def __init__(self,name,kind):
self.name =name
self.kind =kind
defdrink(self):
print('%s在喝水'%self.name)
class Cat(Animal): #派生类
def __init__(self,name,kind,eye_color):
#Animal.__init__(self,name,kind) # 调用父类的方法,要加参数self
super().__init__(name,kind) #调用父类的方法,super不需要加self
self.eye_color = eye_color #派生属性
def catch(self): #派生方法
print('抓老鼠')
子类和父类有同名的方法,如果既要执行子类的方法,也想要执行父类的方法,那么需要在子类的方法中调用父类的方法:
①父类名.方法名(self)
② super().方法名()
super()调用父类的方法,不需要传self,通过父类名调用父类的方法要传self
classFoo:
def __init__(self):
self.func()
deffunc(self):
print('in Foo')
classSon(Foo):
deffunc(self):
print('in Son')
s1 =Son()
'''结果:in Son
'''
self = s1,调用的是Son自己的方法func,所以执行的是子类自身的方法
当self去调用某个方法的时候,不要看self在哪个类里,要看self到底是谁
classFoo:
city = 'ShenZhen'
deffunc(self):
print(self.city)
classSon(Foo):
city = 'HONGKONG'
s =Son()
s.func()
'''结果 HONGKONG
'''
s.func()执行的是父类中的方法,但是调用的属性是Son本身的属性,self就是s,打印的是Son的属性
当self去调用某个方法的时候,不要看self在哪个类里,要看self到底是谁
四、抽象类
抽象类的一个特点是它不能直接被实例化
抽象类的目的就是让别的类继承它并实现特定的抽象方法
抽象基类的一个主要用途是在代码中检查某些类是否为特定类型,实现了特定接口
抽象类的场景:
① 工作中公司有使用抽象类开发的规则
② 看源码,别人的源码使用了抽象类
python使用抽象类的不多
抽象类用来规范代码:
多人开发,复杂的需求、后期的扩展和版本的更新,用新的手段(抽象)来完成规范
抽象类的描述:
抽象类是一个规范,它基本不会实现具体的功能,是由于抽象类不能被实例化
写抽象类:
from abc import ABCMeta,abstractmethod
在这个类创建的时候指定metaclass=ABCMeta
在子类实现的方法加上一个@abstractmethod 装饰器
from abc importABCMeta,abstractmethod
class Payment(metaclass=ABCMeta):
@abstractmethod #abstractmethod 是一个装饰器,装饰器放在函数/类的上一行
defpay(self):
pass
classGooglePay(Payment):
defpay(self,money):
print('支付了%s元'%money)
defpay(obj,money):
obj.pay(money)
G =GooglePay()
a = pay(G,10000)
'''支付了10000元
'''
装饰器一般放在函数或者类的上一行,表示装饰下一行的函数或者类
使用装饰器:
① 继承这个类
② 必须实现这个类(子类创建同名的方法)被abstractmethod 装饰器 装饰的方法
定义了抽象类必须实行抽象方法,否则会报错
五、多继承
不支持的继承的语言:java,C#
支持多继承的语言:C++,python
多继承的弊端:继承的父类有同名的方法
多继承的定义:python一个子类可以调用多个父类的方法
classAnimal:
def __init__(self,name):
self.name =name
classLand(Animal):
defwalk(self):
print('%s 在步行'%self.name)
classSea(Animal):
defswim(self):
print('%s 在游泳'%self.name)
classAir(Animal):
deffly(self):
print('%s 在飞行'%self.name)
classDrogon(Land,Sea,Air):
pass
classFrog(Sea,Land):
pass
drogon = Drogon('东海龙')
drogon.fly()
drogon.swim()
drogon.walk()
'''东海龙 在飞行
东海龙 在游泳
东海龙 在步行
'''
六、接口类
java C# 接口可以被多继承,接口的方法不能有代码。
不允许类的多继承语言,规范继承类必须实现这个方法。
python 中没有接口,抽象类相当于接口
七、新式类和经典类
新式类:python3的版本中,所有的类都是新式类
所有的新式类都有一个默认的父类:object
class Person1:pass #定义类的写法1
class Person2():pass #定义类的写法2
class Person3(object):pass #定义类的写法3
print(Person1.__bases__)
print(Person2.__bases__)
print(Person3.__bases__)
'''(<class 'object'>,)
(<class 'object'>,)
(<class 'object'>,)
'''
定义类的三种写法:
① 类名:
② 类名():
③ 类名(object):
python2.7版本是经典类和新式类并存
继承了object的类就是新式类:
python3中所有的类都是新式类
python2.7既有新式类又又经典类
不主动继承object都是经典类
新式类和经典类之间多继承顺序的区别:
新式类:
所有的多继承关系寻找方法的顺序是---遵循广度优先算法
① 走过的路不能重复
② 所有的点都要走到
新式类中可以使用类.mro() 查看继承顺序
super() 不是单纯的找父类,而是遵循mro顺序的
经典类:
经典类在找父类中方法的过程中---遵循 深度优先
深度优先:①一条路走到底
② 走过的路不走
class A:pass
class B(A):pass
class C(A):pass
class D(B,C):pass
print(D.mro())
'''[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
'''
super() 不是单纯的找父类,而是遵循mro顺序的
classA:
deffunc(self):
print('A')
classB(A):
deffunc(self):
super().func()
print('B')
classC(A):
deffunc(self):
super().func()
print('C')
classD(B,C):
deffunc(self):
super().func()
print('D')
print(D.mro())
D().func()
'''[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
A
C
B
D
'''
新式类与经典类的区别:
新式类:
①所有的多继承关系寻找方法的顺序---遵循广度优先算法
②继承object
③mro方法
④super(): super不是单纯的找父类,而是遵循mro顺序的
经典类:
① python2版本
② 不主动继承object
③ 经典类在找父类中方法的过程中 遵循---深度优先
④ 不提供mro方法和super()