Python编程中对super函数的正确理解和用法解析


Posted in Python onJuly 02, 2016

当在子类需要调用父类的方法时,在python2.2之前,直接用类名调用类的方法,即非绑定的类方法,并把自身对象self作参数传进去。

class A(object): 
  def say(self): 
    print 'I am A' 
 
class B(A): 
  def say(self): 
    print 'I am B' 
    A.say(self) 
 
b = B() 
b.say()

输出

I am B
I am A

这样运作挺好,不过有个问题,当父类改了名字时,就要把这些显式调用父类的一个个更正,子类和父类耦合比较高。
于是python2.2后就推出了super()函数来避免硬编码,不用关心父类名叫什么。
使用super()函数,上面的代码可以写成如下。

class B(A): 
  def say(self): 
    print 'I am B' 
    super(B,self).say()

python3.0后,又做了改良,super()函数不用传参数,即上面的那行代码直接super().say()就行了。

需要注意的问题:

  • super只能用在新式类中。
  • super在多重继承有问题,如果子类继承多个父类,那么super调用第一个父类的方法。
  • 不要混用这两种调用父类方法的方案,要么都用非绑定的类方法,要么都用super。不然可能导致没被调用或者被调用多次。

BUT:
不要一说到 super 就想到父类!super 指的是 MRO 中的下一个类!
一说到 super 就想到父类这是初学者很容易犯的一个错误,也是我当年犯的错误。

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

两个参数 cls 和 inst 分别做了两件事:
1. inst 负责生成 MRO 的 list
2. 通过 cls 定位当前 MRO 中的 index, 并返回 mro[index + 1]
这两件事才是 super 的实质,一定要记住!
MRO 全称 Method Resolution Order,它代表了类继承的顺序。

举个例子:

class Root(object):
  def __init__(self):
    print("this is Root")

class B(Root):
  def __init__(self):
    print("enter B")
    # print(self) # this will print <__main__.D object at 0x...>
    super(B, self).__init__()
    print("leave B")

class C(Root):
  def __init__(self):
    print("enter C")
    super(C, self).__init__()
    print("leave C")

class D(B, C):
  pass

d = D()
print(d.__class__.__mro__)

输出

enter B
enter C
this is Root
leave C
leave B
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)

知道了 super 和父类其实没有实质关联之后,我们就不难理解为什么 enter B 下一句是 enter C 而不是 this is Root(如果认为 super 代表“调用父类的方法”,会想当然的认为下一句应该是this is Root)。流程如下,在 B 的 __init__ 函数中:

super(B, self).__init__()

首先,我们获取 self.__class__.__mro__,注意这里的 self 是 D 的 instance 而不是 B 的

(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)

然后,通过 B 来定位 MRO 中的 index,并找到下一个。显然 B 的下一个是 C。于是,我们调用 C 的 __init__,打出 enter C。

顺便说一句为什么 B 的 __init__ 会被调用:因为 D 没有定义 __init__,所以会在 MRO 中找下一个类,去查看它有没有定义 __init__,也就是去调用 B 的 __init__。

其实这一切逻辑还是很清晰的,关键是理解 super 到底做了什么。

Python 相关文章推荐
python绘图库Matplotlib的安装
Jul 03 Python
初学Python实用技巧两则
Aug 29 Python
python判断字符串是否纯数字的方法
Nov 19 Python
Python多线程、异步+多进程爬虫实现代码
Feb 17 Python
Python使用matplotlib简单绘图示例
Feb 01 Python
全面分析Python的优点和缺点
Feb 07 Python
Python进程间通信Queue消息队列用法分析
May 22 Python
pytorch 求网络模型参数实例
Dec 30 Python
pytorch实现MNIST手写体识别
Feb 14 Python
基于python实现破解滑动验证码过程解析
May 28 Python
FP-growth算法发现频繁项集——构建FP树
Jun 24 Python
Python使用DFA算法过滤内容敏感词
Apr 22 Python
Python中的复制操作及copy模块中的浅拷贝与深拷贝方法
Jul 02 #Python
快速排序的算法思想及Python版快速排序的实现示例
Jul 02 #Python
Python使用functools模块中的partial函数生成偏函数
Jul 02 #Python
Python之父谈Python的未来形式
Jul 01 #Python
举例讲解Python的lambda语句声明匿名函数的用法
Jul 01 #Python
Python内置数据结构与操作符的练习题集锦
Jul 01 #Python
Python设置默认编码为utf8的方法
Jul 01 #Python
You might like
phpmyadmin安装时提示:Warning: require_once(./libraries/common.inc.php)错误解决办法
2011/08/18 PHP
优化PHP代码技巧的小结
2013/06/02 PHP
使用PHP获取汉字的拼音(全部与首字母)
2013/06/27 PHP
PHP脚本监控Nginx 502错误并自动重启php-fpm
2015/05/13 PHP
laravel 解决crontab不执行的问题
2019/10/22 PHP
js中confirm实现执行操作前弹出确认框的方法
2014/11/01 Javascript
基于jQuery的JavaScript模版引擎JsRender使用指南
2014/12/29 Javascript
纯css实现窗户玻璃雨滴逼真效果
2015/08/23 Javascript
微信公众平台开发教程(四) 实例入门:机器人回复(附源码)
2016/12/02 Javascript
浅谈vue中.vue文件解析流程
2018/04/24 Javascript
axios取消请求的实践记录分享
2018/09/26 Javascript
localstorage实现带过期时间的缓存功能
2019/06/28 Javascript
vue 使用鼠标滚动加载数据的例子
2019/10/31 Javascript
JS实现点星星消除小游戏
2020/03/24 Javascript
typescript编写微信小程序创建项目的方法
2021/01/29 Javascript
python使用urlparse分析网址中域名的方法
2015/04/15 Python
Python3.x中自定义比较函数
2015/04/24 Python
Python使用tablib生成excel文件的简单实现方法
2016/03/16 Python
PyQt 线程类 QThread使用详解
2017/07/16 Python
python+pandas生成指定日期和重采样的方法
2018/04/11 Python
python3.x实现base64加密和解密
2019/03/28 Python
在Python中通过threshold创建mask方式
2020/02/19 Python
python GUI库图形界面开发之PyQt5打开保存对话框QFileDialog详细使用方法与实例
2020/02/27 Python
深入了解Python装饰器的高级用法
2020/08/13 Python
python 实现IP子网计算
2021/02/18 Python
美国Jeep配件购物网站:Morris 4×4 Center
2019/05/01 全球购物
英国领先的露营和露营车品牌之一:OLPRO
2019/08/06 全球购物
戴尔荷兰官方网站:Dell荷兰
2020/10/04 全球购物
最新创业融资计划书
2014/01/19 职场文书
护士岗前培训自我评鉴
2014/02/28 职场文书
商业融资计划书
2014/04/29 职场文书
2014年师德师风自我剖析材料
2014/09/27 职场文书
2014年教研员工作总结
2014/12/23 职场文书
质量承诺书格式范文
2015/04/28 职场文书
创业计划书之情侣餐厅
2019/09/29 职场文书
Golang MatrixOne使用介绍和汇编语法
2022/04/19 Golang