注: Python 2.7.x 环境下

今晚搜东西无意中看到这篇Understanding Python super() with __init__() methods.

其实这篇老早就看过了, 不过有一篇很好的回答之前没有注意到.

首先说下super(), 我只在类的单继承时的__init__()中使用过.

注意super只能用在新式类(new-style class)中, 也就是继承自object类对象的子类:

class A(object):
....

以前遇到过一个问题, 排查了半天, 才发现是老式类定义.

传统的super使用方法如:

class Base(object):
def __init__(self, id):
self.id = id class Child(Base):
def __init__(self, id, name):
super(Child, self).__init__(id)
self.name = name

这个是Python2.2之后才支持的特性, 在之前只能:

class Child(Base):
def __init__(self, id, name):
Base.__init__(self, id)
self.name = name

这样做的好处就是不需要显示的在初始化时指明Child的父类名是什么, 在复杂的继承环境下, 以致会牵一发动一身.

不过就像那篇帖子top1的回答里所说:

But the main advantage comes with multiple inheritance

super在多继承这种更复杂的环境下, 才能发挥真正的威力, 这也是python文档中提到的第二个使用场景. 当然至今没遇到过这种复杂环境, 所以没有发言权.


上面扯了一些super的基本情况, 接着该扯下帖子里top2的回答了.

里面提到了这个用法:

super(self.__class__, self).__init__()

关于__class__:

instance.__class__ : The class to which a class instance belongs.

因为前阵子在使用多线程(threading.Thread)时, 写了一个基类, 然后有两个类分别继承自这个基类, 设置线程名就是类名, 这时就用到了__class__:

class base_thread(threading.Thread):
def __init__(self, **kwargs):
threading.Thread.__init__(self)
self.name = self.__class__.__name__

所以对这个比较敏感, 才留意了下这个回答, 没想到却发现了一些坑...

按照帖子里的那个回复:

This unfortunately does not necessarily work if you want to inherit the constructor from the superclass.

例子:

#!/usr/bin/env python
# -*- coding: utf-8 -*- class Polygon(object):
def __init__(self, id):
self.id = id class Rectangle(Polygon):
def __init__(self, id, width, height):
super(self.__class__, self).__init__(id)
self.shape = (width, height) class Square(Rectangle):
pass p = Polygon(10)
print p.id r = Rectangle(5, 1, 2)
print r.id s = Square(20, 2, 4)
print s.id

运行结果:

% python test.py
10
5
Traceback (most recent call last):
File "test.py", line 65, in <module>
s = Square(20, 2, 4)
File "test.py", line 53, in __init__
super(self.__class__, self).__init__(id)
TypeError: __init__() takes exactly 4 arguments (2 given)

执行到Square类时, 报错说应该有4个参数, 但是实际上只有两个.

简化下代码, 并加一些调试输出:

#!/usr/bin/env python
# -*- coding: utf-8 -*- class Polygon(object):
def __init__(self, id):
print('in Polygon, self.__class__ is %s' % self. 大专栏  扯下Python的super()__class__)
self.id = id class Rectangle(Polygon):
def __init__(self, id, width, height):
super(self.__class__, self).__init__(id)
#super(Rectangle, self).__init__(id)
print('in Rectangle, self.__class__ is %s' % self.__class__)
self.shape = (width, height) p = Polygon(10)
print p.id r = Rectangle(5, 1, 2)
print r.id

结果是:

% python test.py
in Polygon, self.__class__ is <class '__main__.Polygon'>
10
in Polygon, self.__class__ is <class '__main__.Rectangle'>
in Rectangle, self.__class__ is <class '__main__.Rectangle'>
5

可以看出来, 在Rectangle初始化时, 通过super调用父类Polygon进行初始化, 而 __class__还是Rectangle.

所以在上一个例子中, Square因为和Rectangle的初始化方法一样, 所以初始化时会调用:

super(Square, self).__init__(id)

即:

Rectangle.__init__(id)

但是实际上Rectangle接收4个参数的初始化, 所以这里报错.

接着考虑, 解决参数个数不一致的问题? 那么就让参数多一致:

#!/usr/bin/env python
# -*- coding: utf-8 -*- class Polygon(object):
def __init__(self, id, width, weight):
print('in Polygon, self.__class__ is %s' % self.__class__)
self.id = id class Rectangle(Polygon):
def __init__(self, id, width, height):
super(self.__class__, self).__init__(id, width, height)
#super(Rectangle, self).__init__(id)
print('in Rectangle, self.__class__ is %s' % self.__class__)
self.shape = (width, height) class Square(Rectangle):
def __init__(self, id, width, height):
super(self.__class__, self).__init__(id, width, height)
#super(Rectangle, self).__init__(id)
print('in Square, self.__class__ is %s' % self.__class__)
self.shape = (width, height) p = Polygon(10, 3, 6)
print p.id r = Rectangle(5, 1, 2)
print r.id s = Square(20, 2, 4)
print s.id

运行报错:

% python test.py
in Polygon, self.__class__ is <class '__main__.Polygon'>
10
in Polygon, self.__class__ is <class '__main__.Rectangle'>
in Rectangle, self.__class__ is <class '__main__.Rectangle'>
5
Traceback (most recent call last):
File "test.py", line 30, in <module>
s = Square(20, 2, 4)
File "test.py", line 18, in __init__
super(self.__class__, self).__init__(id, width, height)
File "test.py", line 11, in __init__
super(self.__class__, self).__init__(id, width, height)
File "test.py", line 11, in __init__ ... File "test.py", line 11, in __init__
super(self.__class__, self).__init__(id, width, height)
File "test.py", line 11, in __init__
super(self.__class__, self).__init__(id, width, height)
File "test.py", line 11, in __init__
super(self.__class__, self).__init__(id, width, height)
RuntimeError: maximum recursion depth exceeded while calling a Python object

在Rectangle的super这一样发生了无限循环.

在Square的super函数里:

super(self.__class__, self).__init__(id, width, height)

相当于:

Rectangle.__init__(id, width, height)

而此时在Retangle的super函数里, __class__还是等于Square, 所以super(self.__class__, self)就是Rectangle自身, 所以在这里发生了死循环.

唯一的做法就是在Square中重定义__init__, 并且和__class__无关.

这块有点绕, 需要理解下.

扯下Python的super()的更多相关文章

  1. Linux环境下Python的安装过程

    Linux环境下Python的安装过程 前言 一般情况下,Linux都会预装 Python了,但是这个预装的Python版本一般都非常低,很多 Python的新特性都没有,必须重新安装新一点的版本,从 ...

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

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

  3. Python: 你不知道的 super

    https://segmentfault.com/a/1190000007426467 Python: 你不知道的 super 在类的继承中,如果重定义某个方法,该方法会覆盖父类的同名方法,但有时,我 ...

  4. python中super的理解(转)

    原文地址:https://www.zhihu.com/question/20040039 针对你的问题,答案是可以,并没有区别.但是这题下的回答我感觉都不够好. 要谈论 super,首先我们应该无视 ...

  5. Python面试题之Python的Super方法

    我们最常见的,可以说几乎唯一能见到的使用super的形式是: class SubClass(BaseClass): def method(self): super(SubClass, self).me ...

  6. Python’s super() considered super!

    如果你没有被Python的super()惊愕过,那么要么是你不了解它的威力,要么就是你不知道如何高效地使用它. 有许多介绍super()的文章,这一篇与其它文章的不同之处在于: 提供了实例 阐述了它的 ...

  7. python的super深入了解(转)

    1.python的继承以及调用父类成员 python子类调用父类成员有2种方法,分别是普通方法和super方法 假设Base是基类 class Base(object): def __init__(s ...

  8. Python中super的用法【转载】

    Python中super的用法[转载] 转载dxk_093812 最后发布于2019-02-17 20:12:18 阅读数 1143  收藏 展开 转载自 Python面向对象中super用法与MRO ...

  9. 算是休息了这么长时间吧!准备学习下python文本处理了,哪位大大有好书推荐的说下!

    算是休息了这么长时间吧!准备学习下python文本处理了,哪位大大有好书推荐的说下!

随机推荐

  1. 实战 迁移学习 VGG19、ResNet50、InceptionV3 实践 猫狗大战 问题

    实战 迁移学习 VGG19.ResNet50.InceptionV3 实践 猫狗大战 问题   参考博客:::https://blog.csdn.net/pengdali/article/detail ...

  2. tensorflow--保存加载模型

    s=mnist.train.next_batch(batch_size)print(xs.shape)print(ys.shape) # #从集合中取全部变量# tf.get_collection() ...

  3. 全面掌握Nginx配置+快速搭建高可用架构 一 Centos7 安装Nginx

    Nginx官网 http://nginx.org/en/linux_packages.html#stable 配置yum 在etc的yum.repos.d目录下新增nginx.repo 将内容copy ...

  4. ArchLinux安装Gnome桌面

    给Arch安装Gnome桌面美化及常用软件配置 一.创建普通用户 1.安装zsh 个人比较喜欢的一个shell,你们可以和我不同 # pacman -S zsh 2.创建用户 kain是我创建用户的名 ...

  5. [GXYCTF2019]Ping Ping Ping

    0x00 知识点 命令执行变量拼接 /?ip=127.0.0.1;a=g;cat$IFS$1fla$a.php 过滤bash用sh执行 echo$IFS$1Y2F0IGZsYWcucGhw|base6 ...

  6. javascript语法规范和良好的变成习惯

    1.1空白和多行书写 1.空白:空格键输入的空白.tab键输入的空白以及回车键输入的空白 2.多行书写,不能将引号内的字符串放到两行,不然容易报错. 1.2点语法 . 点语法表达式由对象开始,接着是一 ...

  7. NBU For Windows 更新后无法启动Java Console

    系统环境:Win Server 2016 备份软件:NBU 8.1   Case :Windows系统默认自动安装系统补丁,重启过程中自动进行了Update,打上了最新的补丁后,NBU Java Co ...

  8. PPT |《Kubernetes的兴起》

    京东云开发者社区技术沙龙--<Cloud Native时代的应用之路与开源创新> Part1-<Kubernetes的兴起> 欢迎点击"链接"了解更多精彩内 ...

  9. ab工具压接口的时候post传参问题

    ab  -n 10000 -c 40 -p  [参数所在文件] -T 'application/json'  http://xxx 以上命令, 压测需要post json格式的参数的api时, 一定注 ...

  10. UML-逻辑架构精化

    向下请求:Facade模式 向上返回:观察者模式 不局限于上图中指定的层使用相应模式,其他层也可以使用. 另外,尽量不要出现“公共业务模块”,设计时尽量做好系统拆分.否则,一旦修改公共代码,可能会影响 ...