• python 面向对象专题(19):基础(10)-继承


    0 ⽬标

    继承的概念
    单继承
    多继承
    ⼦类重写⽗类的同名属性和⽅法
    ⼦类调⽤⽗类的同名属性和⽅法
    多层继承
    super()
    私有属性和私有⽅法

    ⼀. 继承的概念

    拓展1:经典类或旧式类
    不由任意内置类型派⽣出的类,称之为经典类。
    class 类名:
        代码
        ......
    拓展2:新式类
    class 类名(object):
        代码
    Python⾯向对象的继承指的是多个类之间的所属关系,即⼦类默认继承⽗类的所有属性和⽅法,具体如下:
    # 继承:子类默认继承父类的所有属性和方法
    
    
    # 1. 定义父类
    class A(object):
        def __init__(self):
            self.num = 1
    
        def info_print(self):
            print(self.num)
    
    
    # 2. 定义子类 继承父类
    class B(A):
        pass
    
    
    # 3. 创建对象,验证结论
    result = B()
    result.info_print()
    在Python中,所有类默认继承object类,object类是顶级类或基类;其他⼦类叫做派⽣类。

    ⼆. 单继承

    故事主线:⼀个煎饼果⼦⽼师傅,在煎饼果⼦界摸爬滚打多年,研发了⼀套精湛的摊煎饼果⼦的
    技术。师⽗要把这套技术传授给他的唯⼀的最得意的徒弟。
    分析:徒弟是不是要继承师⽗的所有技术?
    # 1. 师父类,属性和方法
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 2. 定义徒弟类,继承师父类
    class Prentice(Master):
        pass
    
    
    # 3. 用徒弟类创建对象,调用实例属性和方法
    daqiu = Prentice()
    
    print(daqiu.kongfu)
    
    daqiu.make_cake()

    三. 多继承

    故事推进:daqiu是个爱学习的好孩⼦,想学习更多的煎饼果⼦技术,于是,在百度搜索到⿊⻢程
    序员,报班学习煎饼果⼦技术。
    所谓多继承意思就是⼀个类同时继承了多个⽗类。
    # 1. 师父类,属性和方法
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 为了验证多继承,添加School父类
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 2. 定义徒弟类,继承师父类 和 学校类
    class Prentice(School, Master):
        pass
    
    
    # 3. 用徒弟类创建对象,调用实例属性和方法
    daqiu = Prentice()
    
    print(daqiu.kongfu)
    
    daqiu.make_cake()
    
    
    # 结论:如果一个类继承多个父类,优先继承第一个父类的同名属性和方法

    四. ⼦类重写⽗类同名⽅法和属性

    故事:daqiu掌握了师⽗和培训的技术后,⾃⼰潜⼼钻研出⾃⼰的独⻔配⽅的⼀套全新的煎饼果⼦技术。
    # 1. 师父类,属性和方法
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 为了验证多继承,添加School父类
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 2. 定义徒弟类,继承师父类 和 学校类, 添加和父类同名的属性和方法
    class Prentice(School, Master):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 3. 用徒弟类创建对象,调用实例属性和方法
    daqiu = Prentice()
    
    print(daqiu.kongfu)
    
    daqiu.make_cake()
    
    # 结论:如果子类和父类拥有同名属性和方法,子类创建对象调用属性和方法的时候,调用到的是子类里面的同名属性和方法
    ⼦类和⽗类具有同名属性和⽅法,默认使⽤⼦类的同名属性和⽅法。

    五. ⼦类调⽤⽗类的同名⽅法和属性

    故事:很多顾客都希望也能吃到古法和⿊⻢的技术的煎饼果⼦。
    # 1. 师父类,属性和方法
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 为了验证多继承,添加School父类
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 2. 定义徒弟类,继承师父类 和 学校类, 添加和父类同名的属性和方法
    class Prentice(School, Master):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
    
        def make_cake(self):
            # 加自己的初始化的原因:如果不加这个自己的初始化,kongfu属性值是上一次调用的init内的kongfu属性值
            self.__init__()
            print(f'运用{self.kongfu}制作煎饼果子')
    
        # 子类调用父类的同名方法和属性:把父类的同名属性和方法再次封装
        def make_master_cake(self):
            # 父类类名.函数()
            # 再次调用初始化的原因:这里想要调用父类的同名方法和属性,属性在init初始化位置,所以需要再次调用init
            Master.__init__(self)
            Master.make_cake(self)
    
        def make_school_cake(self):
            School.__init__(self)
            School.make_cake(self)
    
    
    # 3. 用徒弟类创建对象,调用实例属性和方法
    daqiu = Prentice()
    daqiu.make_cake()
    
    daqiu.make_master_cake()
    
    daqiu.make_school_cake()
    
    daqiu.make_cake()
    六. 多层继承
    故事:N年后,daqiu⽼了,想要把所有技术传承给⾃⼰的徒弟。
    # 1. 师父类,属性和方法
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 为了验证多继承,添加School父类
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    # 2. 定义徒弟类,继承师父类 和 学校类, 添加和父类同名的属性和方法
    class Prentice(School, Master):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
    
        def make_cake(self):
            self.__init__()
            print(f'运用{self.kongfu}制作煎饼果子')
    
        # 子类调用父类的同名方法和属性:把父类的同名属性和方法再次封装
        def make_master_cake(self):
            Master.__init__(self)
            Master.make_cake(self)
    
        def make_school_cake(self):
            School.__init__(self)
            School.make_cake(self)
    
    
    # 步骤:1. 创建类Tusun, 用这个类创建对象;2. 用这个对象调用父类的属性或方法看能否成功
    class Tusun(Prentice):
        pass
    
    
    xiaoqiu = Tusun()
    
    xiaoqiu.make_cake()
    
    xiaoqiu.make_master_cake()
    
    xiaoqiu.make_school_cake()

    七. super()调⽤⽗类⽅法

    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    class School(Master):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
            # 2.1 super()带参数写法
            # super(School, self).__init__()
            # super(School, self).make_cake()
    
            # 2.2 无参数的super
            super().__init__()
            super().make_cake()
    
    
    class Prentice(School):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
    
        def make_cake(self):
            self.__init__()
            print(f'运用{self.kongfu}制作煎饼果子')
    
        # 子类调用父类的同名方法和属性:把父类的同名属性和方法再次封装
        def make_master_cake(self):
            Master.__init__(self)
            Master.make_cake(self)
    
        def make_school_cake(self):
            School.__init__(self)
            School.make_cake(self)
    
        # 需求:一次性调用父类School Master的方法
        def make_old_cake(self):
            # 方法一:如果定义的类名修改,这里也要修改,麻烦; 代码量庞大,冗余
            # School.__init__(self)
            # School.make_cake(self)
            # Master.__init__(self)
            # Master.make_cake(self)
    
            # 方法二:super()
            # 2.1 super(当前类名, self).函数()
            # super(Prentice, self).__init__()
            # super(Prentice, self).make_cake()
    
            # 2.2 无参数super
            super().__init__()
            super().make_cake()
    
    
    daqiu = Prentice()
    
    daqiu.make_old_cake()
    注意:使⽤super() 可以⾃动查找⽗类。调⽤顺序遵循 __mro__ 类属性的顺序。⽐较适合单继承使⽤。
     

    ⼋. 私有权限

    8.1 定义私有属性和⽅法
    在Python中,可以为实例属性和⽅法设置私有权限,即设置某个实例属性或实例⽅法不继承给⼦类。
    故事:daqiu把技术传承给徒弟的同时,不想把⾃⼰的钱(2000000个亿)继承给徒弟,这个时候就
    要为 钱 这个实例属性设置私有权限。
    设置私有权限的⽅法:在属性名和⽅法名 前⾯ 加上两个下划线 __。
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    class Prentice(School, Master):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
            # self.money = 2000000
            # 定义私有属性
            self.__money = 2000000
    
        # 定义私有方法
        def __info_print(self):
            print('这是私有方法')
    
        def make_cake(self):
            self.__init__()
            print(f'运用{self.kongfu}制作煎饼果子')
    
        def make_master_cake(self):
            Master.__init__(self)
            Master.make_cake(self)
    
        def make_school_cake(self):
            School.__init__(self)
            School.make_cake(self)
    
    
    class Tusun(Prentice):
        pass
    
    
    xiaoqiu = Tusun()
    
    # print(xiaoqiu.money)
    # print(xiaoqiu.__money)
    
    xiaoqiu.__info_print()
    注意:私有属性和私有⽅法只能在类⾥⾯访问和修改。
    8.2 获取和修改私有属性值
    在Python中,⼀般定义函数名 get_xx ⽤来获取私有属性,定义 set_xx ⽤来修改私有属性值。
    class Master(object):
        def __init__(self):
            self.kongfu = '[古法煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    class School(object):
        def __init__(self):
            self.kongfu = '[黑马煎饼果子配方]'
    
        def make_cake(self):
            print(f'运用{self.kongfu}制作煎饼果子')
    
    
    class Prentice(School, Master):
        def __init__(self):
            self.kongfu = '[独创煎饼果子技术]'
            # 定义私有属性
            self.__money = 2000000
    
        # 定义函数:获取私有属性值 get_xx
        def get_money(self):
            return self.__money
    
        # 定义函数:修改私有属性值 set_xx
        def set_money(self):
            self.__money = 500
    
        # 定义私有方法
        def __info_print(self):
            print('这是私有方法')
    
        def make_cake(self):
            self.__init__()
            print(f'运用{self.kongfu}制作煎饼果子')
    
        def make_master_cake(self):
            Master.__init__(self)
            Master.make_cake(self)
    
        def make_school_cake(self):
            School.__init__(self)
            School.make_cake(self)
    
    
    class Tusun(Prentice):
        pass
    
    
    xiaoqiu = Tusun()
    
    print(xiaoqiu.get_money())
    
    xiaoqiu.set_money()
    
    print(xiaoqiu.get_money())

    九. 总结

    继承的特点
    ⼦类默认拥有⽗类的所有属性和⽅法
    ⼦类重写⽗类同名⽅法和属性
    ⼦类调⽤⽗类同名⽅法和属性
    super()⽅法快速调⽤⽗类⽅法
    私有权限
    不能继承给⼦类的属性和⽅法需要添加私有权限
    语法
    class 类名():
         # 私有属性
         __属性名 =# 私有⽅法
         def __函数名(self):
             代码
     
     
  • 相关阅读:
    mybatis源码解读(二)——构建Configuration对象
    mybatis源码解读(一)——初始化环境
    JDK1.8源码(七)——java.util.HashMap 类
    JDK1.8源码(六)——java.util.LinkedList 类
    JDK1.8源码(五)——java.util.ArrayList 类
    JDK1.8源码(四)——java.util.Arrays 类
    JDK1.8源码(三)——java.lang.String 类
    JDK1.8源码(二)——java.lang.Integer 类
    JDK1.8源码(一)——java.lang.Object类
    Java的深拷贝和浅拷贝
  • 原文地址:https://www.cnblogs.com/qiu-hua/p/14969360.html
Copyright © 2020-2023  润新知