C3算法 和 super
一. Python的继承 多继承
子类继承父类.
继承是为了节省开发时间.提高开发效率 代码得到了重(chong)用 一个类可以拥有多个父类
lass shen_xian: # 定义一个神仙类
def fei(self): # 神仙类有一个方法 可以飞
print('神仙都会飞')
class Monkey: # 定义一个猴子类
def chitao(self): # 猴子有一个方法 可以吃桃子
print('猴子喜欢吃桃')
class SunWuKong(Monkey,shen_xian): # 对象继承猴子和神仙的类型
pass
sxz = SunWuKong() # 实例化一个孙悟空对象
sxz.chitao() # 会吃桃子
sxz.fei() # 会飞
多继承用起来简单 但是存在一个问题 当父类中出现重名方法时 就涉及到查找父类方法中的问题 也即MRO(method resolution order)
二. 经典类的MRO
在Python2中存在两种类
经典类 : 在Python 2.2 之前 一直使用的是 经典类
新式类 : 在2.2之后 出现了新式类 特点是 基类的根是object
Python 3
只有新式类 如果基类谁都不继承 那么会默认继承object
经典类的MRO 深度优先遍历 例如 快递员送鸡蛋 :

肯定是按照123456顺序来送 即每次都是最左边 找完撤回到分叉口继续往里找(从左往右,一条道跑到黑,然后撤回继续一条道跑到黑) 即深度优先遍历 如果 142356 就是广度优先遍历
三. 新式类的MRO, C3(重点, 难点) 可以通过 类名.__mro__ 获取到类的mro信息
即print(类名.__mro__) 就可以获取到
Python中的新式类的mro都是用 c3算法来完成的
笔试的时候肯定会考
新式类中摒弃了(部分). C3算法
如果继承关系没有菱形继承(深度优先)
如果有菱形:使用C3算法来计算MRO
三. super
super()可以执行 mro中的下一个父类的方法 通常super()有两个使用的地方 :
1 . 可以访问父类的构造方法
2 . 当子类方法想调用父类(mro)中的方法
先看第一种
class Foo:
def __init__(self,a,b,c):
self.a = a
self.b = b
self.c = c
class Bar(Foo):
def __init__(self,a,b,c,d):
super().__init__(a,b,c)
self.d = d
b = Bar(1,2,3,4)
print(b.__dict__) # {'a': 1, 'b': 2, 'c': 3, 'd': 4}
这样就方便了子类 不需要写那么多了
第二种
class Foo:
def func1(self):
super().func1() # 此时找的是mro顺序中下一个类的func1()方法 实际到这就会报错 根本不会运行
# 因为 Foo 中没有func1方法 继承的obj中也没有func1()方法
print('我的老家,就住在这个屯')
class Bar:
def func(self):
print('你的老家,不在这个屯')
class Ku(Foo,Bar):
def func1(self):
super().func1()
print('它的老家,不知道在哪个屯')
k = Ku() # 先看mro Ku , Foo, Bar ,object
k.func1() k2 = Foo() # 此时的mro Foo , object 都没有func1 因此会报错 如果这样写一开始就会报错 就不会运行到这才报错
k2.func1()
五. 一道面试题
# mro + super 面试题
class Init:
def __init__(self,v):
print('init')
self.val = v
class Add2(Init):
def __init__(self,val):
print('Add2')
super(Add2,self).__init__(val)
print(self.val)
self.val += 2
class Mult(Init):
def __init__(self,val):
print('Mult')
super(Mult,self).__init__(val)
self.val *= 5
class HaHa(Init):
def __init__(self,val):
print('哈哈')
super(HaHa,self).__init__(val)
self.val /= 5
class Pro(Add2,Mult,HaHa):
pass
class Incr(Pro):
def __init__(self, val):
super(Incr, self).__init__(val)
self.val += 1
p = Incr(5)
print(p.val)
c = Add2(2)
print(c.val)
结论 : 不管super()写在哪. 在哪执行,一定先找到mro列表.根据mro列表的顺序往下找 否则都是错的.
C3算法 和 super的更多相关文章
- MRO和C3算法
本节主要内容: 1.python多继承 2.python经典类的MRO 3.python新式类的MRO,C3算法 4.super() 一.python多继承 在python中类与类之间可以有继承关系, ...
- day21 MRO和C3算法
核能来袭 --MRO和C3算法 1. python的多继承 2.python经典类的MRO 3.python新式类的MRO, C3算法 4.super 是什么鬼? 一.python的多继承 在前面的学 ...
- python学习 day20 (3月27日)----(单继承多继承c3算法)
继承: 提高代码的重用性,减少了代码的冗余 这两个写法是一样的 Wa('青蛙').walk() #青蛙 can walk wa = Wa('青蛙') wa.walk() #青蛙 can walk 1. ...
- MRO C3算法 super的运用
-------------态度决定成败,无论情况好坏,都要抱着积极的态度,莫让沮丧取代热心.生命可以价值极高,也可以一无是处,随你怎么去选择.# --------------------------- ...
- python 面向对象(六)MRO C3算法 super
########################总结################ 面向对象回顾 类:对某一个事物的描述,对一些属性和方法的归类 class 类名: var=123#类变量 def ...
- day19 MRO C3算法 super()
1. MRO(Method Resolution Order):方法解析顺序,主要用于在多继承时判断调的属性的路径(来自于哪个类). 1.Python语言包含了很多优秀的特性,其中多重继承就是其中之一 ...
- super之mro列表牵引出c3算法
目录 一:super的使用 二:super之mro列表牵引出c3算法 三:mro列表总结使用 一:super的使用 class Person(object): def __init__(self, n ...
- python小兵 面向对象继承super和c3算法
python多继承 在前面的学习过程中. 我们已经知道了Python中类与类之间可以有继承关系. 当出现了x是一种y的的时候. 就可以使⽤继承关系. 即"is-a" 关系. 在继承 ...
- 手推C3算法
C3算法规则 -- 每一个类的继承顺序都是从基类向子类看 -- 形成一个指向关系的顺序[当前类] + [父类的继承顺序] -- 进行一个提取 -- 如果一个类出现从左到右的第一个顺序上,并且没有出现在 ...
随机推荐
- C++中的一类临时对象
类名(参数名)这样的对象是临时对象,不能取地址,不能被引用,不过可以给同类型的其他对象赋值,该临时对象定以后可以进行一次操作,然后立即销毁. 当我们定义一个对象以后并不想立即给它赋初值,而是以后给它赋 ...
- Action层, Service层 和 Dao层的功能区分
Action/Service/DAO简介: Action是管理业务(Service)调度和管理跳转的. Service是管理具体的功能的. Action只负责管理,而Service负责实施. DAO ...
- STM32 C++编程 004 Adc (数模转换)类
使用 C++ 语言给 STM32 编写一个 Adc 类 我使用的STM32芯片:STM32F103ZET6 我们使用的STM32库版本:V3.5.0 注意: 想学习本套 STM32 C++编程 的专栏 ...
- 算法Sedgewick第四版-第1章基础-015一stack只保留last指针
/************************************************************************* * * A generic queue, impl ...
- swing JCheckBox 更换复选框样式
Java Swing - 如何自定义JCheckBox复选标记图标 摘自 https://www.w3cschool.cn/java/codedemo-484050311.html import ja ...
- CodeForces 703C Chris and Road (简单几何)
题意:有一个n边形的汽车向以速度v向x轴负方向移动,给出零时时其n个点的坐标.并且有一个人在(0,0)点,可以以最大速度u通过w宽的马路,到达(0,w)点.现在要求人不能碰到汽车,人可以自己调节速度. ...
- redis系列:哨兵
1 简介 Sentinel(哨兵)是Redis 的高可用性解决方案:通过哨兵可以创建一个当主服务器出现故障时自动将从服务器升级为主服务器的一个分布式系统.解决了主从复制出现故障时需要人为干预的问题. ...
- Java50道经典习题-程序42 求数字
题目:809*??=800*??+9*??+1其中??代表两位数,若有这样得数,求??代表的两位数 public class Prog42{ public static void main(Strin ...
- WndProc和hook区别
1)WndProc函数作用:主要在程序中拦截并处理系统消息和自定义消息 比如:windows程序会产生很多消息,比如你单击鼠标,移动窗口都会产生消息.这个函数就是默认的消息处理函数.你可以重载这个函数 ...
- 记一次RSA非对称算法的排坑经历
Map<String,Object> encryParam = new HashMap<>(5); encryParam.put("connectorUrl" ...