python---面对对象的三大特征

一、三大特征

  面对对象的三大特征:继承、多态和封装,继承在面向对象类的创建中非常常见。

1、继承

  为什么会有继承,当我们要创建一个新类的时候,发现他有很多属性或者反法都和我们另一个类的方法相同,这时我们就可以用继承。Python中继承分为单继承和多继承。

复制代码
class Animal:
    def __init__(self):
        self.name = name
        self.kind = kind
        self.food = food
        self.language = language
    def yell(self):
        print('%s叫'%self.language)
    def eat(self):
        print('吃%s'%self.foode)
    def drink(self):
        print('喝水')
复制代码

上面已经有了一个Animal类,属性有名字、类型、食物、语言属性,方法有叫,吃,喝水.这时我叫生成一个狗类和猫类,你会发现在狗猫类中也有这些行为和方法这样我们就可以用到继承

复制代码
class Animal:
    def __init__(self,name,kind,food,language):
        self.name = name
        self.king = kind
        self.food = food
        self.language = language
    def yell(self):
        print('%s叫'%self.language)
    def eat(self):
        print('吃%s'%self.food)
    def drink(self):
        print('喝水')

class Cat(Animal):
    def look_after(self):
        print('看家')

class Dog(Animal):
    def catch_mouse(self):
        print('抓老鼠')

Tom = Cat('汤姆','家猫','猫粮','喵喵')
Tom.yell()  #喵喵叫
Tom.eat()  #吃猫粮

jd = Dog('京东狗','白狗','狗粮','汪汪')
jd.yell()  #汪汪叫
jd.eat()  #吃狗粮
复制代码

Cat(Animale)这就是继承的语法,在括号里面的叫做父类或者超类基累,而Cat叫做派生类,派生类可以调用父类的所有方法以及属性,同时遵循以下规则:

子类可以使用父类中的名字(变量和方法)

子类在父类的基础上又新创建了自己需要的方法和属性

父类有的子类没有 - 子类对象直接调用 就会直接执行父类的方法

父类有的子类也有 - 子类对象调用 直接执行子类中的方法

复制代码
class Animal:
    def __init__(self,name,kind,food,language):
        self.name = name
        self.king = kind
        self.food = food
        self.language = language
    def yell(self):
        print('%s叫'%self.language)
    def eat(self):
        print('吃%s'%self.food)
    def drink(self):
        print('喝水')

class Cat(Animal):
    def look_after(self):
        print('看家')
    def yell(self):
        print('子类方法')

class Dog(Animal):
    def catch_mouse(self):
        print('抓老鼠')

Tom = Cat('汤姆','家猫','猫粮','喵喵')
Tom.catch_mouse()  # catch_mouse方法只有子类中有调用子类方法
Tom.yell()   #yell方法在子类和父类中都有,会优先找子类的
Tom.eat()  #子类中没有eat方法,调用父类的eat方法
复制代码

二、派生

  前面讲到子类可以调用父类的属性,也就是父类中init的特征属性,那么子类可以有自己的特征属性吗,当然是可以的。需要注意的是,一旦重新定义了自己的属性且与父类重名,那么调用新增的属性时,就以子类自己的特征属性为主了。

复制代码
class Animal:
    def __init__(self,name,kind,food,language):
        self.name = name
        self.kind = kind
        self.language = language
    def eat(self):
        print('吃%s'%self.food)

class Cat(Animal):
    def __init__(self,eyecolor):
        self.eyecolor = eyecolor

Tom = Cat('蓝色')
print(Tom.eyecolor)    #蓝色
print(Tom.name)    #报错
复制代码

这时大家可能会问为什么Tom.name会报错呢,还是遵循这样的原则:父类有的子类也有 - 子类对象调用 直接执行子类中的方法  

大家注意 __init()__这个函数是不是子类和父类在都有,这时子类会优先执行子类自己的__init()__方法,这样子类的属性就只有eyecolor而已你去调用对象Tom去找时也只会在子类自己的__init()方法中去找,所以这样子类的属性中就只有子类自己的属性了。

其实子类自己初始化__init__的时候将父类的属性添加到子类的属性中去就可以了:

复制代码
class Animal:
    def __init__(self,name,kind,food,language):
        self.name = name
        self.kind = kind
        self.language = language
    def eat(self):
        print('吃%s'%self.food)

class Cat(Animal):
    def __init__(self,name,kind,food,language,eyecolor):
        self.eyecolor = eyecolor
        super().__init__(name,kind,food,language)    
    # super().__init__(name,kind,food,language)
    #或者替换成Animal.__init__(self,name,kind,food,language)

Tom = Cat('汤姆','家猫','猫粮','喵喵','蓝色')
print(Tom.name)    #汤姆
复制代码
但需记得的是将父类的属性添加进来后,子类实例化的参数也要随着属性的增多添加。
总结;

 当子类当中有要被调用的方法的时候,子类的对象会直接选择子类中的方法、变量,父类中的方法不会被自动执行

 如果我们既想要执行子类的方法,也想要执行父类的方法,那么需要在子类的方法中调用父类的方法:
 父类名.方法名(self,...)
 super().方法名(...)
帮助我们在子类中调用父类中的同名方法

三、多继承

  多继承,在python中一个子集是可以继承多个父级的

 

复制代码

class Animal:
def __init__(self,name):
self.name = name
def fly(self):
print('%s在飞'%self.name)
def walk(self):
print('%s在走路'%self.name)
def swim(self):
print('%s在游泳'%self.name)

obj = Animal('狗')
obj.walk()  #狗在走路
obj.swim()  #狗在游泳
obj.fly()    #狗在飞
obj = Animal('鸟')
obj.fly()  #鸟在飞
复制代码

 

看样看出实例化出来的对象可以调用父类所有的方法,连狗都能飞了!这样肯定不行,那么如何让狗调用不了飞的方法呢,其实我们可以更具体,将飞的方法变成一个类,走的也变成一个类,而动物类只负责生成属性:

复制代码
class Animal:
    def __init__(self,name):
        self.name = name

class FlyAnimal(Animal):
    def fly(self):
        print('%s在飞' % self.name)
class WalkAnimal(Animal):
    def walk(self):
        print('%s在走路'%self.name)
class SwimAnimal(Animal):
    def swim(self):
        print('%s在游泳'%self.name)

class Tiger(SwimAnimal,WalkAnimal):
    pass

class Swan(SwimAnimal,WalkAnimal,FlyAnimal):
    pass

class Parrot(FlyAnimal,WalkAnimal):
    def talk(self):
        print('%s说话了'%self.name)

swan = Swan('天鹅')
swan.fly()
swan.walk()
复制代码

 

 

继承小结:

  

python到底是如何实现继承的,对于你定义的每一个类,python会计算出一个方法解析顺序(MRO)列表,这个MRO列表就是一个简单的所有基类的线性顺序列表,例如

>>> F.mro() #等同于F.__mro__
[<class '__main__.F'>, <class '__main__.D'>, <class '__main__.B'>, <class '__main__.E'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]

 

为了实现继承,python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。
而这个MRO列表的构造是通过一个C3线性化算法来实现的。我们不去深究这个算法的数学原理,它实际上就是合并所有父类的MRO列表并遵循如下三条准则:
1.子类会先于父类被检查
2.多个父类会根据它们在列表中的顺序被检查
3.如果对下一个类存在两个合法的选择,选择第一个父类

 

posted @   Kingfan  阅读(301)  评论(0编辑  收藏  举报
努力加载评论中...
点击右上角即可分享
微信分享提示