python-面向对象-继承

继承的概念

生活中的继承,一般指的是子女继承父辈的财产。

拓展经典类和新式类

拓展1:经典或旧式类
不由任意内置内置型派生出的类,称之为经典类。

class 类名:
	代码
	......

拓展2:新式类

class 类名(object):
	代码

python面向对象的继承指的是多个类之间的所属关系,即子类默认继承父类的所有属性的方法。

  • 体验继承
# 继承:子类默认继承父类的所有属性和方法
# 定义父类
from unittest import result


class A(object):
    def __init__(self) -> None:
        self.num = 1
    
    def info_print(self):
        print(self.num)

# 定义子类 继承父类
class B(A):
    pass

# 创建对象,验证结论
result = B()
result.info_print() # 1

在python中,所有类默认继承object类,object类是顶级类或基类;其他子类叫做派生类。

单继承

故事主线:一个煎饼果子老师傅,在煎饼果子界摸爬滚打多年,研发了一套精湛的摊煎饼果子的技术。师父要把这套技术传授给他唯一的最得意的徒弟。

分析:徒弟是不是要继承师父的所有技术?

# 1. 师父类:属性和方法
class Master():
    def __init__(self) -> None:
        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():
    def __init__(self) -> None:
        self.kongfu = '[古法煎饼果子配方]'

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 2. 定义培训学校的类
class School():
    def __init__(self) -> None:
        self.kongfu = '[溏心蛋果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 3. 徒弟继承师父和培训学校的类
class Prentice(School,Master):
    pass


daqiu = Prentice()
print(daqiu.kongfu) # [溏心蛋果子配方]
daqiu.make_cake() # 运用[溏心蛋果子配方]制作煎饼果子

结论:如果一个类继承多个父类,优先继承第一个父类的同名属性和方法

子类重写父类同名方法和属性

故事:daqiu掌握了师父和培训的技术后,自己潜心钻研出自己的独门配方的一套新的煎饼果子技术。

# 1. 定义师父的类
class Master():
    def __init__(self) -> None:
        self.kongfu = '[古法煎饼果子配方]'

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 2. 定义培训学校的类
class School():
    def __init__(self) -> None:
        self.kongfu = '[溏心蛋果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 3. 徒弟独创配方
class Prentice(School,Master):
    def __init__(self) -> None:
        self.kongfu = '[独创煎饼果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')


daqiu = Prentice()
print(daqiu.kongfu) # [独创煎饼果子配方]
daqiu.make_cake() # 运用[独创煎饼果子配方]制作煎饼果子

结论:如果子类和父类拥有同名属性和方法,子类创建对象调用属性和方法的时候,调用到的是子类里面的同名属性和方法。

拓展_mro顺序

得到继承关系
测试:


print(Prentice.__mro__)
# (<class '__main__.Prentice'>, <class '__main__.School'>, <class '__main__.Master'>, <class 'object'>)

子类调用父类的同名方法和属性

故事:一部分顾客喜欢吃古法煎饼果子,也有一部分喜欢溏心蛋煎饼果子。

# 1. 定义师父的类
class Master():
    def __init__(self) -> None:
        self.kongfu = '[古法煎饼果子配方]'

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 2. 定义培训学校的类
class School():
    def __init__(self) -> None:
        self.kongfu = '[溏心蛋果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 3. 徒弟独创配方
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) # 再次调用初始化的原因:这里想要调用父类同名属性和方法,属性在init初始化位置,需要再次调用init
        Master.make_cake(self) # 如果不加self会报错
    
    def make_school_cake(self):
        School.__init__(self)
        School.make_cake(self)

# 4. 子类调用父类的同名方法
daqiu = Prentice()
daqiu.make_master_cake() # 运用[古法煎饼果子配方]制作煎饼果子

多层继承

故事:daqiu老了,想要把所有技术传承给自己的徒弟。

# 1. 定义师父的类
class Master():
    def __init__(self) -> None:
        self.kongfu = '[古法煎饼果子配方]'

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 2. 定义培训学校的类
class School():
    def __init__(self) -> None:
        self.kongfu = '[溏心蛋果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 3. 徒弟独创配方
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) # 再次调用初始化的原因:这里想要调用父类同名属性和方法,属性在init初始化位置,需要再次调用init
        Master.make_cake(self) # 如果不加self会报错
    
    def make_school_cake(self):
        School.__init__(self)
        School.make_cake(self)

# 二代徒弟类
class Tusun(Prentice):
    pass

xiaoqiu = Tusun()
xiaoqiu.make_cake() # 运用[独创煎饼果子配方]制作煎饼果子
xiaoqiu.make_master_cake() # 运用[古法煎饼果子配方]制作煎饼果子
xiaoqiu.make_school_cake() # 运用[溏心蛋果子配方]制作煎饼果子

super()方法作用

调用父类的属性和方法。

父类.方法调用 VS super()方法

父类.方法

上面已经写过很多了,这里总结一下:

  1. 如果前面的类名修改,后面子类使用父类里面也要修改。
  2. 代码量庞大,比较冗余

super()方法

# 1. 定义师父的类
class Master():
    def __init__(self) -> None:
        self.kongfu = '[古法煎饼果子配方]'

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 2. 定义培训学校的类
class School():
    def __init__(self) -> None:
        self.kongfu = '[溏心蛋果子配方]'
    
    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')

# 3. 徒弟独创配方
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)

    def make_old_cake(self):
        super(Prentice,self).__init__()
        super(Prentice,self).make_cake()

daqiu = Prentice()
daqiu.make_old_cake() # 运用[溏心蛋果子配方]制作煎饼果子

这里的例子比较抽象,如果想深入了解,可以私聊我,很愿意解答

私有权限

定义所有属性和方法

在Pyhton中,可以为实例属性和方法设置私有权限,即设置某个实例属性或方法不继承给子类。

故事:daqiu把技术传承给徒弟的同时,不想把自己的钱(200000000)继承给徒弟,这个时候就要为这个实例属性设置私有权。

设置是由权限的方法:
在属性名和方法名前加上两个下划线。
体验:

class Prentice():
    def __init__(self) -> None:
        self.kongfu = '[独家煎饼果子]'
        self.__money = 200000000

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')
    
    def print_info(self):
        print(self.kongfu)
        print(self.__money)

class Tusun(Prentice):
    pass

daqiu = Prentice()
daqiu.print_info()
''' 
[独家煎饼果子]
200000000
'''
xiaoqiu = Tusun()
Tusun.print_info() # 报错

注意:是由属性和私有方法只能在类里面访问和修改

获取和修改私有属性值

在python中,一边定义函数名get_xx用来获取是由属性,定义set_xx用来修改私有属性值。

class Prentice():
    def __init__(self) -> None:
        self.kongfu = '[独家煎饼果子]'
        self.__money = 200000000

    def make_cake(self):
        print(f'运用{self.kongfu}制作煎饼果子')
    
    # 调用私有属性
    def print_info(self):
        print(self.kongfu)
        print(self.__money)
    
    # 获取私有属性
    def get_money(self):
        return self.__money

    # 修改私有属性
    def set_money(self,money):
        self.__money = money

class Tusun(Prentice):
    pass

xiaoqiu = Tusun()
money = xiaoqiu.get_money()
print(xiaoqiu.get_money()) # 200000000
xiaoqiu.set_money(500)
print(xiaoqiu.get_money()) # 500

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

溏心蛋*

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值