13 继承-方法重写-多态

13 继承和方法的重写

目标
  • 单继承
  • 多继承

面向对象三大特性

  1. 封装根据职责将属性和方法封装到一个抽象的类中
  2. 继承实现代码的重用,相同的代码不需要重复的编写
  3. 多态不同的对象调用相同的方法,产生不用的执行结果,增加代码的灵活性
单继承

继承的概念、语法和特点

继承的概念:子类拥有父类的所有方法和属性

继承的语法
claa 类名(父类名):
	pass
  • 子类继承自父类,可以直接享受父类中已封装好的方法,不需要再次开发
  • 子类中应该根据职责,封装子类特有的属性和方法
class Animal:
    def eat(self):
        print("吃")

    def drink(self):
        print("喝")

    def run(self):
        print("跑")

    def sleep(self):
        print("睡")


class Dog(Animal):
    def bark(self):
        print("叫")


wangcai = Dog()
wangcai.eat()
wangcai.drink()
wangcai.run()
wangcai.sleep()
wangcai.bark()
专业术语
  • Dog 类是 Animal 类的子类,Animal 类是 Dog 类的父类,Dog 类从 Animal 类继承
  • Dog 类是 Animal 类的派生类,Animal 类是 Dog 类的基类,Dog 类从 Animal 类派生
继承的传递性
  • C 类从 B 类继承,B 类又从 A 类继承
  • 那么C 类就具有 B 类和 A 类的所有属性和方法

子类拥有父类以及父类的父类中封装的所有属性和方法

class Animal:
    def eat(self):
        print("吃")

    def drink(self):
        print("喝")

    def run(self):
        print("跑")

    def sleep(self):
        print("睡")


class Dog(Animal):
    def bark(self):
        print("叫")


class xiaotianquan(Dog):
    def fly(self):
        print("飞")


xtq = xiaotianquan()
xtq.fly()
xtq.bark()
xtq.eat()
方法的重写
  • 子类拥有父类的所有方法和属性
  • 子类继承自父类,可以直接享受父类中已经封装好的方法,不需要再次开发

应用场景

  • 当父类的方法实现不能满足子类需求时,可以对方法进行重写(override)

重写父类方法有两种情况:

  1. 覆盖父类的方法
  2. 对父类方法进行扩展
覆盖父类的方法
  • 如果在开发中,父类方法实现和子类的方法实现,完全不同
  • 就可以使用覆盖的方式,在子类中重新编写父类的方法实现

具体的实现方式,就相当于在子类中定义了一个和父类同名的方法并且实现

重写之后,在运行时,只会调用子类中重写的方法,而不会调用父类封装的方法

class Animal:
    def eat(self):
        print("吃")

    def drink(self):
        print("喝")

    def run(self):
        print("跑")

    def sleep(self):
        print("睡")


class Dog(Animal):
    def bark(self):
        print("叫")


class xiaotianquan(Dog):
    def fly(self):
        print("飞")

    def bark(self):
        print("叫的跟神一样...")


xtq = xiaotianquan()
# 如果在子类中重写了父类的方法
# 在使用子类调用方法时,会调用子类中重写的方法,而不会调用父类的方法
xtq.bark()
对父类方法进行扩展
  • 如果在开发中,子类的方法实现中包含父类的方法实现
    • 父类原本封装的方法实现是子类方法的一部分
  • 就可以使用扩展的方式
    1. 在子类中重写父类的方法
    2. 在需要的位置使用super().父类方法 来调用父类方法的执行
    3. 代码其他的位置针对子类的需求,编写子类特有的代码实现

关于 super

  • pythonsuper 是一个特殊的类
  • super() 就是使用 super 类创建出来的对象
  • 最常使用的场景就是在重写父类方法时,调用在父类中封装的方法实现
class Animal:
    def eat(self):
        print("吃")

    def drink(self):
        print("喝")

    def run(self):
        print("跑")

    def sleep(self):
        print("睡")


class Dog(Animal):
    def bark(self):
        print("叫")


class xiaotianquan(Dog):
    def fly(self):
        print("飞")

    # 在子类中重写父类的方法
    def bark(self):
        print("叫的跟神一样...")

        # 使用super(). 调用原本在父类封装的方法
        super().bark()

        # 增加子类其他代码
        print("123")


xtq = xiaotianquan()
# 如果在子类中重写了父类的方法
# 在使用子类调用方法时,会调用子类中重写的方法,而不会调用父类的方法
xtq.bark()

父类的私有属性和方法

私有属性、方法是对象的隐私,不对外公开,外界以及子类都不能直接访问

私有属性、方法通常用于做一些内部的事情

  • 在子类对象方法中,不能访问父类的私有属性,同样也不能调用父类的私有方法

  • 子类对象可以在自己的方法内部,访问父类的公有属性和公有方法

  • 子类对象可以通过父类的公有方法间接访问到私有属性或私有方法

class A:
    def __init__(self):
        self.__num = 100

    def __test(self):
        print("私有方法")

    def test(self):
        print("调用私有属性 %d" % self.__num)
        self.__test()


class B(A):
    pass


b = B()
b.test()
多继承

概念:

  • 子类可以拥有多个父类,并且具有所有父类的属性和方法
  • 列如:孩子会继承父亲和母亲的特性
class A:
    def test(self):
        print("test方法")


class B:
    def demo(self):
        print("demo方法")


class C(A, B):
    """多继承可以让子类对象同时具有多个父类的属性和方法"""
    pass


c = C()
c.test()
c.demo()
多继承使用注意事项
  • 如果不同的父类中存在同名的方法,子类对象在调用方法时,会调用哪一个父类中的方法呢?

    提示:开发时,应该尽量避免这种容易产生混淆的情况!如果父类之间存在同名的属性或者方法,应该尽量避免使用多继承

类					
class A:		   class B:
test(self):			test(self):
demo(self):			demo(self):	
c子类继承A,B两个父类
C
A:
B:
python 中的 MRO

——方法搜索顺序

  • python 中针对类提供了一个内置属性 __mro__ 可以查看方法搜索顺序
  • mromethod resolution order,主要用于在多继承时判断方法、属性的调用路劲
print (C.__mro__)

输出结果

class A:
    def demo(self):
        print("A---demo方法")

    def test(self):
        print("A---test方法")


class B:
    def demo(self):
        print("B---demo方法")

    def test(self):
        print("B---test方法")


class C(A, B):
    """多继承可以让子类对象同时具有多个父类的属性和方法"""
    pass


c = C()
# 确定C类对象调用方法的顺序
print(C.__mro__)

(<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <class 'object'>)
  • 在搜索方法时,是按照 __mro__ 的输出结果从左至右的顺序查找的
  • 如果在当前类中找到方法,就直接执行,不在搜索
  • 如果没有找到,就查找下一个类中是否有对应的方法,如果找到,就直接执行,不在搜索
  • 如果找到最后一个类,还没有找到方法,程序报错
新式类与旧式类

objectpython 为所有对象提供的基类,提供一些内置的属性和方法,可以使用 dir 函数查看

  • 新式类:以 object 为基类的类,推荐使用
  • 旧式类:不以 object 为基类的类,不推荐使用
  • python3.x中定义类时,如果没有指定父类,会默认使用object 作为该类的父类(基类)
    • python3.x 中定义的类都是新式类
  • python2.x 中定义的类,如果没有指定父类,则不会以 object 作为基类

新式类和旧式类在多继承时——会影响到方法的搜索顺序

为了保证编写的代码能够同时在 python2.xpython3.x 运行!

今后在定义父类时,如果没有父类,建议统一继承自 object

class 类名(object):
	pass
多态

面向对象三大特性

  1. 封装: 根据职责将属性和方法封装到一个抽象的类中
    • 定义类的准则
  2. 继承: 实现代码的重用,相同的代码不需要重复的编写
    • 设计类的技巧
    • 子类针对自己特有的需求,编写特定的代码
  3. 多态: 不同的子类对象调用相同的父类方法,产生不同的执行结果
    • 多态可以增加代码的灵活度
    • 以继承和重写父类方法为前提
    • 是调用方法的技巧,不会影响到类的内部设计
多态案例演练

需求

  1. Dog 类中封装方法 game
    • 普通狗只是简单的玩耍
  2. 定义 XiaoTianQuan 继承 Dog ,并且重写 game 方法
    • 哮天犬需要在天上玩耍
  3. 定义 Person 类,并且封装一个和狗玩的方法
    • 在方法内部、直接让狗对象调用 game 方法
class Dog(object):
    def __init__(self, name):
        self.name = name

    def game(self):
        print("%s 普通的玩耍..." % self.name)


class XiaoTianQuan(Dog):
    def game(self):
        print("%s 飞到天上去玩耍..." % self.name)


class Person(object):
    def __init__(self, name):
        self.name = name

    def game_with_dog(self, dog):
        print("%s 和 %s 快乐的玩耍..." % (self.name, dog.name))

        # 让狗玩耍
        dog.game()


# 创建狗对象
# wangcai = Dog("旺财")
wangcai = XiaoTianQuan("旺财")

# 创建人对象
xiaoming = Person("小明")

# 让人跟狗玩
xiaoming.game_with_dog(wangcai)

案例小结

  • Person 类中只需要让狗对象调用 game 方法,而不关心具体是什么狗
    • game 方法是在 Dog 父类中定义的
  • 在程序执行时,传入不同的狗对象实参,就会产生不同的执行效果
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值