对象的 __init__() 方法在 python 中做了啥? [复制]
Posted
技术标签:
【中文标题】对象的 __init__() 方法在 python 中做了啥? [复制]【英文标题】:What does object's __init__() method do in python? [duplicate]对象的 __init__() 方法在 python 中做了什么? [复制] 【发布时间】:2012-01-26 13:22:19 【问题描述】:在阅读 OpenStack 的代码时,我遇到了这个问题。
一个名为'Service'的类继承了基类'object',然后在Service的__init__()
方法中,调用了object的__init__
。相关代码如下:
类定义:
class Service(object):
以及Service的init方法定义:
def __init__(self, host, binary, topic, manager, report_interval=None,
periodic_interval=None, *args, **kwargs):
并在 Service 的 init 中调用 super(此处为“对象”):
super(Service, self).__init__(*args, **kwargs)
我不明白最后一个电话,object.__init__()
它实际上是做什么的?
谁能帮忙?
【问题讨论】:
【参考方案1】:简短的回答是 object.__init__() 方法除了检查是否没有传入任何参数之外什么都不做。有关详细信息,请参阅the source。
当在 Service 的实例上调用时,super() 调用将委托给 object.__init__() 并且不会发生任何事情。
然而,当调用 Service 的子类的实例时,事情会变得更有趣。 super() 调用可以潜在地委托给 object 以外的某个类,该类是实例的父类,但不是 Service的父类>。有关其工作原理以及为何有用的详细信息,请参阅博客文章 Python's Super Considered Super!
下面的例子(有点做作)展示了 Service 的子类如何导致 Service 中的 super 调用被定向到另一个类称为颜色:
class Service(object):
def __init__(self, host, binary, topic, manager, report_interval=None,
periodic_interval=None, *args, **kwargs):
print 'Initializing Service'
super(Service, self).__init__(*args, **kwargs)
class Color(object):
def __init__(self, color='red', **kwargs):
print 'Initializing Color'
self.color = color
super(Color, self).__init__(**kwargs)
class ColoredService(Service, Color):
def __init__(self, *args, **kwds):
print 'Initializing Colored Service'
super(ColoredService, self).__init__(*args, **kwds)
c = ColoredService('host', 'bin', 'top', 'mgr', 'ivl', color='blue')
在示例中,初始化按以下顺序进行:
-
初始化彩色服务
初始化服务
初始化颜色
初始化对象——除了参数检查什么都不做
【讨论】:
参考Python's Super Considered Super 有很大帮助。谢谢!ColoredService
构造函数允许例如foo='bar'
被通过,但这最终会以TypeError: object.__init__() takes no parameters
失败。那么在super
调用Service
和Color
类中传递*args
和**kwargs
有什么意义呢?您的示例在没有它们的情况下也可以正常工作(并且会更加健壮)。
@ekhumoro 该评论应该针对 OpenStack 的作者。 *args
部分来自他们的代码,调用者必须尊重该签名。我刚刚添加了 Color 和 ColoredService 来展示 MRO 如何从 Service 链接到 Color 到 object .【参考方案2】:
super()
并不总是返回父类的代理。相反,它返回MRO中的下一个类的代理。在单继承中,MRO 和继承链没有区别。在多重继承中,MRO 可能会导致 other 继承链上的类。
【讨论】:
但是这里没有多重继承,因为Service直接继承自对象 @can.: 但是从Service
继承了什么?
我看过关于MRO的文章,我认为因为Service直接继承自object,所以super()在这里没有歧义。从 Service 继承的 Whaterver 没有任何区别。
@can.:转到标题为“C3 方法解析顺序”部分的示例并查看B
的 MRO。【参考方案3】:
object.__init__()
实际上并没有做任何事情,除了 super()
调用应该包括在内,即使一个类只有 object
作为超类。
“super”的一个大问题是它听起来会导致 要调用的方法的超类副本。这根本不是 在这种情况下,它会导致 MRO 中的下一个方法被调用 (...) 如果唯一的超类是“对象”,则省略对 super(...).init 的调用, 毕竟 object.init 什么都不做!然而,这是 非常不正确。这样做会导致其他类的 init 方法 不会被调用。
http://fuhm.net/super-harmful/
【讨论】:
但请注意,如果通过*args
和**kwargs
传递任何内容,OP 的示例代码将失败(除非它使用 python 2.5 或更早版本运行)。
你们两个能详细解释一下你们的想法吗? @ekhumoro以上是关于对象的 __init__() 方法在 python 中做了啥? [复制]的主要内容,如果未能解决你的问题,请参考以下文章