Python 面向对象 之 多继承 MRO

关于子类的继承.

在子类初始化的时候, 是需要手动调用父类的初始化方法来说明的, 语法为: **父类名.__ init__(self, ...)**

多继承 - Bug

这里主要演示一个, 多继承会出现的问题, 其实也不算bug吧, 原理我也没有仔细去推敲过哈, 先露出来看看吧.

class 父亲:
    def __init__(self, name):
        self.name = name
        print("i am you father...")

class 长子(父亲):
    def __init__(self, name, age):
        self.age = age
        父亲.__init__(self, name)
        print(" i am eldest son")

class 次子(父亲):
    def __init__(self, name, gender):
        self.gender = gender
        父亲.__init__(self, name)
        print("i am the little son")

class 长孙(长子, 次子):
    """收养的哦"""
    def __init__(self, name, age, gender):
        长子.__init__(self, name, age)
        次子.__init__(self, name, gender)
        print("i am the sweet heart")

if __name__ == '__main__':

    king = 长孙("小王子", 23, "F")
i am you father...
i am eldest son

i am you father...  # 多执行了一次 ???

i am the little son
i am the sweet heart

卧槽. ....父类被执行了2次, 这感觉是浪费内存了呀, 嗯..有些紧张....

为啥会出现调用2次父类呢? 或者 如何只让调用一次就好呢 ? 分析一波上面这一段代码

长子.__init__(self, name, age) --> class 长子(父亲) --> class 父亲 -> print("father...")
次子.__init__(self, name, gender) --> class 次子(父亲) --> class 父亲 -> print("father...")

因此, 被调用了2次是没有问题的, 但这就带来一个新的优化问题: 浪费内存, 因为重复开辟空间了呀.

更一般地描述, 即在多继承的情况下, 父类的属性可能会出现被多次执行的情况, 这里仅仅是一个简单的case, 如果非常复杂的结果, 则会浪费非常多的内存和反复调用.

多继承 MRO

MRO - 引入

MRO 是 Pyhton 用来解决上面的这种, 会产生重复开辟空间的问题. 解决的办法是将 复杂结构上的所有类 全部映射到一个线性顺序上 (线性表), 这个顺序就是 MRO 顺序.

MRO表, 事先就调用好了, 按表的顺序调用, 从右到左

print(长孙.__mro__)

# output

(<class '__main__.长孙'>, <class '__main__.长子'>, <class '__main__.次子'>, <class '__main__.父亲'>, <class 'object'>)

这个线性表, 我从执行的角度来看, 就是个 , 父类在栈底, 小王子在栈顶 (Top) , 而搜索的顺序, 兄弟们, 定睛一看, 这一层, 一层第搜索,.... 感觉这就是 树的广度优先算法 呀.

MRO - super

官方已经把这个问题给解决了, 方式就是使用关键字 super

class 父亲:
    def __init__(self, name):
        self.name = name
        print("i am you father...")

class 长子(父亲):
    def __init__(self, name, age, *args):
        self.age = age
        super().__init__(name, *args)
        print(" i am eldest son")

class 次子(父亲):
    def __init__(self, name, gender):
        self.gender = gender
        super().__init__(name)
        print("i am the little son")

class 长孙(长子, 次子):
    """收养的哦"""

    def __init__(self, name, age, gender):
        super().__init__(name,  age, gender)
        print("i am the sweet heart")

if __name__ == '__main__':
    king = 长孙("小王子", 23, "F")
    print(长孙.__mro__)
i am you father...
i am the little son
i am eldest son
i am the sweet heart

(<class '__main__.长孙'>, <class '__main__.长子'>, <class '__main__.次子'>, <class '__main__.父亲'>, <class 'object'>)

小结

  • 多继承会产生重复调用的问题, 浪费内存空间
  • MRO顺序是Python将类映射到一个线性表中来调用, 有序的, 能避免重复调用, 搜索算法目测是广度优先
  • super 的理解和用法

原文地址:https://www.cnblogs.com/chenjieyouge/p/12246113.html

时间: 2024-10-10 13:39:24

Python 面向对象 之 多继承 MRO的相关文章

18.Python面向对象之:继承

一:什么面向对象的继承? 比较官方的说法就是: 继承(英语:inheritance)是面向对象软件技术当中的一个概念. 如果一个类别A“继承自”另一个类别B,就把这个A称为“B的子类别”, 而把B称为“A的父类别”也可以称“B是A的超类”.继承可以使得子类别 具有父类别的各种属性和方法,而不需要再次编写相同的代码.在令子 类别继承父类别的同时,可以重新定义某些属性,并重写某些方法,即覆 盖父类别的原有属性和方法,使其获得与父类别不同的功能.另外,为子类 别追加新的属性和方法也是常见的做法. 一般

python面向对象--封装,继承,多态

1.创建类 class ClassName: ''' 定义类 ''' def __init__(self,name,age):#self代表类的实例,而不是类本身 ''' 类初始化函数 :param name:姓名 :param age: 年龄 ''' self.name=name self.age=age def Class_method(self): ''' 类中的方法 :return: ''' pass 2.类实例化,创建类的对象 c_name1=ClassName('zhangsan',

47、Python面向对象中的继承有什么特点?

继承的优点: 1.建造系统中的类,避免重复操作. 2.新类经常是基于已经存在的类,这样就可以提升代码的复用程度. 继承的特点: 1.在继承中基类的构造(__init__()方法)不会被自动调用,它需要在其派生类的构造中亲自专门调用.有别于C# 2.在调用基类的方法时,需要加上基类的类名前缀,且需要带上self参数变量.区别于在类中调用普通函数时并不需要带上self参数 3.Python总是首先查找对应类型的方法,如果它不能在派生类中找到对应的方法,它才开始到基类中逐个查找.(先在本类中查找调用的

python 面向对象三大特性--继承

一.什么是继承 继承是一种创建新类的方式,在python中,新建的类可以继承一个或者多个父类,父类又可以称为基类活超类,新建的类称为派生类或子类. python中类的继承分为:单继承和多继承 class ParentClass1: #定义父类 pass class ParentClass2: #定义父类 pass class SubClass1(ParentClass1): #单继承,基类是ParentClass,派生类是SubClass pass class SubClass2(ParentC

python面向对象--私有和继承

一. 私有属性和私有方法 应用场景 在实际开发中,对象的某些属性或方法可能只希望在对象的内部使用,而不希望在外部被访问到 私有属性 就是 对象 不希望公开的属性 (属性即类里面的变量) 私有方法 就是 方法 不希望公开的方法 定义方式 在定义属性或方法时,在属性名或者方法名前添加两个下划线,定义的就是私有属性或方法,只能在类里面用. class My: def test(self): self.__password = 123456 #私有属性 def say(self): print('pas

Python高级语法-多继承MRO相关-多继承顺序(4.5.1)

目录 1.说明 2.代码 关于作者 @ 1.说明 使用类的魔法方法__mro__ 可以查看他的父类调用顺序 还有调用父类的构造方法的时候,使用super调用,里面有C3算法支持,不会重复调用相同的祖先类 2.代码 class Parent(object): def __init__(self,name,*args,**kwargs): print("Parent的init开始调用") self.name = name print("Parent的init结束调用")

Python高级语法-多继承MRO相关-args和kwargs(4.5.2)

目录 1.说明 2.代码 关于作者 @ 1.说明 args数据类型为元组 kwargs数据类型为字典 一般传入方法中使用遍历去得到值 这个传入参数的顺序没有特殊的要求 当你自定义的参数传完以后,写了名字的就给kwargs,没有名字的给args 2.代码 def test(*args,**kwargs): for i in args: print(i) for i in kwargs: print(kwargs[i]) if __name__ == '__main__': test(1,2,3,3

Python 面向对象(下)

本篇博客承接自Python 面向对象(上) 四. 继承,实现,依赖,关联,聚合,组合 待补充 参考资料: https://www.cnblogs.com/chinxi/p/7349768.html 五. 特殊成员 待补充 参考资料: 类中的特殊成员方法 待补充 参考资料: https://www.cnblogs.com/chinxi/p/7349768.html 五. 特殊成员 待补充 参考资料: 类中的特殊成员方法 六. issubclass,type,isinstence各自的用法和区别 1

python面向对象super函数

python面向对象super函数 待办 python面向对象可以多继承,同时集成的两个类继承自同一个父类的时候初始化问题,提前定义初始顺序,如果有相同的就按顺序初始化,不要把super理解成父类 https://blog.csdn.net/robinjwong/article/details/48369615 原文地址:https://www.cnblogs.com/lishikai/p/12382320.html