关于python:为什么__new__和__init__在挑选超类时表现得有所不同?

Why do __new__ and __init__ appear to behave differently when punting to a superclass?

我在阅读有关元类的内容时遇到了以下代码,尽管我不知道这种区别是否特定于元类,我怀疑它不是:

1
2
3
4
5
6
7
8
9
10
class MetaBase(type):
    def __new__(mcl, name, bases, nmspc):
        print('MetaBase.__new__
'
)
        return super(MetaBase, mcl).__new__(mcl, name, bases, nmspc)

    def __init__(cls, name, bases, nmspc):
        print('MetaBase.__init__
'
)
        super(MetaBase, cls).__init__(name, bases, nmspc)

注意,super().__init__()调用省略了第一个参数。我猜想它是隐式传递的,因为它对super()返回的任何类调用一个方法。这就是我通常看到的构造此类调用的方式,尽管它们通常涉及普通类上的self,而不是元类上的cls/mcl。

不过,super().__new__()调用显式地传递mcl。我不明白为什么。签名在我看来是一样的。

我很困惑。在每种情况下,super()是否返回不同的内容?这里发生了什么,当我覆盖其他魔法方法时,我应该被它咬吗?

[编辑:有人建议这是这个问题的副本,描述了他们的不同功能。虽然在这里的一些例子中也显示了相同的差异,但我没有看到任何说明它存在的原因或它是否是__new__所独有的。]


The super().__new__() call passes mcl explicitly, though. I don't
understand why. The signatures look the same to me.

__new__方法是一种奇怪的方法,因为它是一种静态方法,但它有特殊的情况,因此您不必将其声明为静态方法。作为静态方法意味着您必须显式传递所需的任何变量(在本例中是元类)。

相比之下,__init__与其他常规方法不同。当您从实例中查找它时,该实例将自动作为第一个参数传入。无论您是直接使用cls.__init__(name, bases, nmspc)调用父对象,还是使用super()和super(MetaBase, cls).__init__(name, bases, nmspc)调用父对象,这都是有效的。这两个参数都作为第一个参数传入CLS。这就是绑定方法在普通类和元类中的工作方式。是否使用super()并不重要。

因此,唯一奇怪的情况是__new__,因为它是一个静态方法,根据定义,它不具有自动预处理行为。

should I expect to be bitten by this when overriding other magic
methods?

静态方法比较少见。在魔法方法中,我能想到的只有__new__。所以,你应该是安全的——)