gpt4 book ai didi

python - 为什么我不能在 python3 中子类化元组?

转载 作者:太空狗 更新时间:2023-10-29 23:57:33 25 4
gpt4 key购买 nike

让我们先说这个问题 you should use __new__ instead of __init__ for subclassing immutable objects .

话虽如此,让我们看下面的代码:

class MyTuple(tuple):
def __init__(self, *args):
super(MyTuple, self).__init__(*args)

mytuple = MyTuple([1,2,3])

这在 python2 中有效,但在 python3 中我得到:

Traceback (most recent call last):
File "tmp.py", line 5, in <module>
mytuple = MyTuple([1,2,3])
File "tmp.py", line 3, in __init__
super(MyTuple, self).__init__(*args)
TypeError: object.__init__() takes no parameters

为什么会这样? python3 有什么变化?

最佳答案

Python 3 改变了 object.__new__object.__init__ 在两者都被覆盖时对参数的 react 。如果一个类重写(或继承重写的方法)object.__init__object.__new__object.__init__object. __new__ 如果他们收到任何多余的参数,将抛出异常。在 Python 2 中,这会给出 DeprecationWarning(默认情况下被抑制)。

tuple 没有自己的 __init__。它继承了 object.__init__,因此您实际上是将一堆参数传递给 object.__init__object.__init__ 不接受。 Python 2 给你一个(被抑制的)警告,而 Python 3 让它成为一个错误。

代码中的注释很好地解释了 object.__init__object.__new__ 对额外参数的微妙处理:

/* You may wonder why object.__new__() only complains about arguments
when object.__init__() is not overridden, and vice versa.

Consider the use cases:

1. When neither is overridden, we want to hear complaints about
excess (i.e., any) arguments, since their presence could
indicate there's a bug.

2. When defining an Immutable type, we are likely to override only
__new__(), since __init__() is called too late to initialize an
Immutable object. Since __new__() defines the signature for the
type, it would be a pain to have to override __init__() just to
stop it from complaining about excess arguments.

3. When defining a Mutable type, we are likely to override only
__init__(). So here the converse reasoning applies: we don't
want to have to override __new__() just to stop it from
complaining.

4. When __init__() is overridden, and the subclass __init__() calls
object.__init__(), the latter should complain about excess
arguments; ditto for __new__().

Use cases 2 and 3 make it unattractive to unconditionally check for
excess arguments. The best solution that addresses all four use
cases is as follows: __init__() complains about excess arguments
unless __new__() is overridden and __init__() is not overridden
(IOW, if __init__() is overridden or __new__() is not overridden);
symmetrically, __new__() complains about excess arguments unless
__init__() is overridden and __new__() is not overridden
(IOW, if __new__() is overridden or __init__() is not overridden).

However, for backwards compatibility, this breaks too much code.
Therefore, in 2.6, we'll *warn* about excess arguments when both
methods are overridden; for all other cases we'll use the above
rules.

*/

关于python - 为什么我不能在 python3 中子类化元组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41390372/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com