开始之前,先出一道题:

 #super函数探讨
class A(object):
def __init__(self):
print 'A.__init__' class B(A):
def __init__(self):
super(B, self).__init__()
print 'B.__init__' class C(A):
def __init__(self):
super(C, self).__init__()
print 'C.__init__' class D(B, C):
def __init__(self):
super(D, self).__init__()
print 'D.__init__' d = D()

上面的运行结果是什么?

是下面的结果吗?

A.__init__
B.__init__
D.__init__

正确答案:

A.__init__
C.__init__
B.__init__
D.__init__

有没有疑惑?super()函数不是调用指定类的父类的方法吗!打印了A.__init__下一句为什么是C.__init__呢?

根本原因是:

super 和父类没有实质性的关联

首先,我们知道新式类采用广度优先算法,我们来看一下上面的继承关系:

那么,Python是如何实现继承的,继承顺序又是由谁决定的呢? 对于你定义的每一个类而已,Python会计算出一个所谓的方法解析顺序(MRO Method Resolution Order)列表。类的继承顺序就是由这个MRO决定的

MRO通过class.__mro__来查看,我们来打印一下上面例子中的MRO:

print D.__mro__
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <type 'object'>)

注意__mro__是类的属性,实例没有该属性

这个MRO列表就是一个简单的所有基类的线性顺序表。为了实现继承,Python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。

而这个MRO列表的构造是通过一个C3线性化算法来实现的。 它实际上就是合并所有父类的MRO列表并遵循如下三条准则:

1)子类会先于父类被检查

2)多个父类会根据它们在列表中的顺序被检查

3)如果对下一个类存在两个合法的选择,选择第一个父类

好像还是没明白为什么例子中,打印了A.__init__下一句为什么是C.__init__呢?

我们使用一个函数来解释一下super的原理:

def super(cls, inst):
mro = inst.__class__.mro()
return mro[mro.index(cls) + 1]

其中,cls 代表类,inst 代表实例,上面的代码做了两件事:

1)获取 inst 的 MRO 列表

2)查找 cls 在当前 MRO 列表中的 index, 并返回它的下一个类,即 mro[index + 1]

当你使用 super(cls, inst) 时,Python 会在 inst 的 MRO 列表上搜索 cls 的下一个类。

是不是有一种豁然开朗的赶脚!让我们回到例子中,这里我画出了整个流程;

从上面的流程图就可以看出打印的顺序是对的!

了解了super的原理,那么也就可以理解下面这段有趣的代码了:

1)执行下面代码

 class A(object):
def go(self):
print 'A go'
super(A, self).go() a = A()
a.go()

会报错:

AttributeError: 'super' object has no attribute 'go'

2)执行下面代码:

 class A(object):
def go(self):
print 'A go'
super(A, self).go() class B(object):
def go(self):
print 'B go' class C(A, B):
pass c = C()
c.go()

不会报错,结果为:

A go
B go

充分说明了super 和父类没有实质性的关联

另外,我们想出了super以外,还有一种直接调用父类方法的方法,如下:

 #super函数探讨
class A(object):
def __init__(self):
print 'A.__init__' class B(A):
def __init__(self):
# super(B, self).__init__()
A.__init__(self)
print 'B.__init__' class C(A):
def __init__(self):
# super(C, self).__init__()
A.__init__(self)
print 'C.__init__' class D(B, C):
def __init__(self):
# super(D, self).__init__()
B.__init__(self)
C.__init__(self)
print 'D.__init__' d = D()

为什么不用这种方法呢?我们运行一下,看一下,结果为:

A.__init__
B.__init__
A.__init__
C.__init__
D.__init__

很明显,A的构造函数运行了两次,这不是我们所希望的;所以还是用super吧!

super函数没有那么简单-super原理剖析的更多相关文章

  1. Python中的super函数,你熟吗?

    摘要:经常有朋友问,学 Python 面向对象时,翻阅别人代码,会发现一个 super() 函数,那这个函数的作用到底是什么? 本文分享自华为云社区<Python中的super函数怎么学,怎么解 ...

  2. python基础----多态与多态性、super函数用法、继承原理

    一.多态与多态性                                                                        ㈠多态: 多态指的是一类事物有多种形态, ...

  3. 由Python的super()函数想到的

    python-super *:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !im ...

  4. Python super() 函数的概念和例子

    概念: super() 函数是用于调用父类(超类)的一个方法. super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,但是如果使用多继承,会涉及到查找顺序(MRO).重 ...

  5. Python super() 函数

    super() 函数是用于调用父类(超类)的一个方法. super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,但是如果重定义某个方法,该方法会覆盖父类的同名方法,但有时 ...

  6. python super()函数详解

    引言: 在类的多继承使用场景中,重写父类的方法时,可能会考虑到需要重新调用父类的方法,所以super()函数就是比较使用也很必要的解决方法: 文章来源: http://www.cnblogs.com/ ...

  7. Python面试题之Super函数

    这是个高大上的函数,在python装13手册里面介绍过多使用可显得自己是高手 23333. 但其实他还是很重要的. 简单说, super函数是调用下一个父类(超类)并返回该父类实例的方法. 这里的下一 ...

  8. python 中 super函数的使用

    转载地址:http://python.jobbole.com/86787/ 1.简单的使用 在类的继承中,如果重定义某个方法,该方法会覆盖父类的同名方法,但有时,我们希望能同时实现父类的功能,这时,我 ...

  9. Python3中的super()函数详解

    关于Python3中的super()函数 我们都知道,在Python3中子类在继承父类的时候,当子类中的方法与父类中的方法重名时,子类中的方法会覆盖父类中的方法, 那么,如果我们想实现同时调用父类和子 ...

随机推荐

  1. Invert a binary tree 翻转一棵二叉树

    Invert a binary tree 翻转一棵二叉树 假设有如下一棵二叉树: 4  / \   2    7  / \   / \ 1  3 6  9翻转后: 4     /    \    7 ...

  2. 51nod_1677:treecnt

    题目是求一棵n节点树中对于C(n,k)颗子树,每棵子树为在n个节点中选不同的k个节点作为树的边界点,这样的所有子树共包含多少条边. 问题可以转化一下,对每一条边,不同的子树中可能包含可能不包含这条边, ...

  3. Spring初学

    一.spring体系结构spring核心组件 1.Beans(包装应用程序自定义对象Object,Object中存有数据) 2.Core (资源加载,资源抽象,建立维护与bean之间的一些关系所需的一 ...

  4. (一)SQL关联查询的使用技巧 (各种 join)

    ---恢复内容开始--- (一)SQL关联查询的使用技巧 (各种 join) 这几天因为工作的时候,发现自己的sql语句基础不是很好,特意研究了一下,发现sql语句真的是博大精深,sql语句不仅是要查 ...

  5. Java之戳中痛点 - (6)避免类型自动转换,例如两个整数相除得浮点数遇坑

    先来看一个例子: package com.test; public class calculate { /** * 光速30万公里/秒 */ public static final int LIGHT ...

  6. Angular02 将数据添加到组件中

    准备:已经搭建好angular-cli环境.知道如何创建组件 一.将一个数据添加到组件中 1 创建一个新的组件 user-item 2 将组件添加到静态模板中 3 为组件添加属性,并利用构造器赋值 4 ...

  7. (转)Spring的单例模式底层实现

    单例模式也属于创建型模式,所谓单例,顾名思义,所指的就是单个实例,也就是说要保证一个类仅有一个实例. 单例模式有以下的特点: ① 单例类只能有一个实例 ② 单例类必须自己创建自己的唯一实例 ③ 单例类 ...

  8. Spring Ioc-依赖注入的几种方式

    一 setter方法注入 配置文件如下: <bean id="helloAction" class="org.yoo.action.SpringSetterHell ...

  9. MySql 事务与锁

    事务介绍 首先,什么是事务?事务就是一段sql 语句的批处理,但是这个批处理是一个atom(原子),不可分割,要么都执行,要么回滚(rollback)都不执行. MySQL 事务主要用于处理操作量大, ...

  10. WordPress中函数钩子hook的作用及基本用法

    WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它.钩子分类 钩子分为两种,一 ...