python面向对象之继承与重写

一、举实例来引入继承

定义一个类 Animal(动物)
这个类中需要两个方法:run() sleep()
定义一个类 Dog(狗)
这个类中需要三个方法:run() sleep() bark()

分析:有一个类,能够实现我们需要的大部分功能,但是不能实现全部功能
如何能让这个类来实现全部的功能呢?

方式一:直接修改原来的类
这种方式的弊端:修改起来会比较麻烦,并且会违反OCP原则 OCP(开闭原则,对扩展开放,对修改是关闭的)

class Animal:
    def run(self):
        print('动物会跑~~~')

    def sleep(self):
        print('动物睡觉~~~')

    # 第一种方式:直接修改这个类
    def bark(self):
       print('动物嚎叫~~~')

方式二:直接新建一个类
这种方式的弊端:创建一个新的类比较麻烦,并且需要大量的进行复制粘贴,会出现大量的重复性代码。

# 第二种方式:直接新建一个类
class Dog:
    def run(self):
        print('狗会跑~~~')

    def sleep(self):
        print('狗睡觉~~~')

    def bark(self):
        print('汪汪汪~~~')

方式三:使用继承的方式

class Animal:
    def run(self):
        print('动物会跑~~~')

    def sleep(self):
        print('动物睡觉~~~')

    # 第一种方式:直接修改这个类
    # def bark(self):
    #     print('动物嚎叫~~~')


# 第三种方式是从Animal类中来继承它的属性和方法
class Dog(Animal):
    def bark(self):
        print('汪汪汪~~~')

    def run(self):
        print('狗跑~~~~')


d = Dog()
d.run()
d.sleep()
d.bark()

二、什么是继承?

  • 继承是面向对象三大特性之一
    - 通过继承我们可以使一个类获取到其他类中的属性和方法
    - 在定义类时,可以在类名后的括号中指定当前类的父类(超类、基类、super)
    子类(衍生类)可以直接继承父类中的所有的属性和方法

继承的作用或者说是好处:

通过继承可以直接让子类获取到父类的方法或属性,避免编写重复性的代码,并且也符合OCP原则
所以我们经常需要通过继承来对一个类进行扩展。

三、isinstance()和issubclass()

isinstance()用来检查一个对象是否是一个类的实例
如果这个类是这个对象的父类,也会返回True
所有的对象都是object的实例

issubclass() 检查一个类是否是另一个类的子类

class Animal:
    def run(self):
        print('动物会跑~~~')

    def sleep(self):
        print('动物睡觉~~~')

    # 第一种方式:直接修改这个类
    # def bark(self):
    #     print('动物嚎叫~~~')


# 第三种方式是从Animal类中来继承它的属性和方法
class Dog(Animal):
    def bark(self):
        print('汪汪汪~~~')

    def run(self):
        print('狗跑~~~~')


class Hashiqi(Dog):
    def fan_sha(self):
        print('我是一只傻傻的哈士奇')


d = Dog()
r = isinstance(d, Dog)
r3 = isinstance(d, Hashiqi)
r2 = isinstance(d, Animal)
print(r)  # True
print(r2)  # # True
print(r3)  # False

print(issubclass(Animal, Dog)) # False
print(issubclass(Animal, object)) # True

在创建类时,如果省略了父类,则默认父类为object
object是所有类的父类,所有类都继承自object
class Person(object):
pass

四、子类也可继承父类中特殊方法

比如子类可继承父类的魔术方法,init

class Animal:
    def __init__(self, name):
        self._name = name

    def run(self):
        print('动物会跑~~~')

    def sleep(self):
        print('动物睡觉~~~')

    @property
    def name(self):
        return self._name

    @name.setter
    def name(self, name):
        self._name = name


class Dog(Animal):
    # 子类Dog继承父类Animal的init方法
    def __init__(self, name, age):
        # 希望可以直接调用父类的__init__来初始化父类中定义的属性
        #   super()可以用来获取当前类的父类,
        #   并且通过super()返回对象调用父类方法时,不需要传递self
        super().__init__(name)
        self._age = age

    def run(self):
        print('狗跑~~~~')

    @property
    def age(self):
        return self._age

    @age.setter
    def age(self, age):
        self._age = age


d = Dog('旺柴', 20)
print(d.name)  # 旺柴
print(d.age)  # 20

五、多重继承

多重继承指的是:我们可以为一个类同时指定多个父类

在Python中是支持多重继承的,也就是我们可以为一个类同时指定多个父类
可以在类名的()后边添加多个类,来实现多重继承
多重继承,会使子类同时拥有多个父类,并且会获取到所有父类中的方法
在开发中没有特殊的情况,应该尽量避免使用多重继承,因为多重继承会让我们的代码过于复杂
如果多个父类中有同名的方法,则会现在第一个父类中寻找,然后找第二个,然后找第三个。。。
前边父类的方法会覆盖后边父类的方法。

class A(object):
    def test(self):
        print('A中的test方法~~~')


class B(object):
    def test(self):
        print('B中的test()方法~~')

    def test2(self):
        print('BBB')


class C(A, B):
    pass


# 类名.__bases__ 这个属性可以用来获取当前类的所有父类
print(C.__bases__)  # (<class '__main__.B'>, <class '__main__.A'>)
print(B.__bases__)  # (<class 'object'>,)

# C 同时继承A和B类,A、B类中有相同的方法,那么当子类C调用两个父类相同的方法时,应该打印谁的结果呢
c = C()
c.test()  
# class C(B, A): 打印结果就是:B中的test()方法~~
# class C(A, B): 打印结果就是:A中的test方法~~~ 取决于,写继承多个父类时写的顺序

六、重写

子类中如果有和父类同名的方法,则通过子类实例去调用方法时,
会调用子类的方法而不是父类的方法,这个特点我们成为叫做方法的重写(覆盖,override)

class Animal:
    def run(self):
        print('动物会跑~~~')

    def sleep(self):
        print('动物睡觉~~~')


class Dog(Animal):
    def bark(self):
        print('汪汪汪~~~')
    
    # 如果子类继承了父类,并也定义了和父类同名的方法,那么就叫方法的重写
    def run(self):
        print('狗跑~~~~')


d = Dog()
d.run()  # 狗跑~~~~

当我们调用一个对象的方法时,
会优先去当前对象中寻找是否具有该方法,如果有则直接调用
如果没有,则去当前对象的父类中寻找,如果父类中有则直接调用父类中的方法,
如果没有,则去父类的父类中寻找,以此类推,直到找到object,如果依然没有找到,则报错。

class A(object):
    def test(self):
        print('AAA')

class B(A):
    def test(self):
        print('BBB')

class C(B):
    def test(self):
        print('CCC')   

# 创建一个c的实例
c = C()
c.test() # 当C的实例调用test方法时,会先去找C ,再去找B,再去找A,再去找object,直到找不着,报错。
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值