我正在尝试将python 3.6的新__init_subclass__
功能(PEP 487)与abc
模块一起使用。它似乎没有工作。以下代码:
from abc import ABCMeta
class Initifier:
def __init_subclass__(cls, x=None, **kwargs):
super().__init_subclass__(**kwargs)
print('got x', x)
class Abstracted(metaclass=ABCMeta):
pass
class Thingy(Abstracted, Initifier, x=1):
pass
thingy = Thingy()
运行时产生以下内容:
Traceback (most recent call last):
File "<filename>", line 10, in <module>
class Thingy(Abstracted, Initifier, x=1):
TypeError: __new__() got an unexpected keyword argument 'x'
如果Abstracted不使用ABCMeta
元类,那么一切正常。
此错误相当具有弹性,例如,以下代码仍会因类似错误而失败(可能是因为元类'__new__
在类实例化时运行,而父类'__new__
没有运行直到对象实例化。
from abc import ABCMeta
class Initifier:
def __new__(cls, name, bases, dct, x=None, **kwargs):
return super().__new__(cls, name, bases, dct, **kwargs)
def __init_subclass__(cls, x=None, **kwargs):
super().__init_subclass__(**kwargs)
print('got x', x)
class Abstracted(metaclass=ABCMeta):
pass
class Thingy(Initifier, Abstracted, x=1):
pass
thingy = Thingy()
任何人都可以确认这是Python 3.6 abc
模块和/或__init_subclass__
实现中的错误吗? (我可能使用__init_subclass__
错误。)有没有人有解决方法?
答案 0 :(得分:6)
这是abc.ABCMeta
中的一个错误,因为__init_subclass__
的设计中存在瑕疵。我建议报告。
现在几乎所有存在的元类都应该将意外的关键字参数传递到super().__new__
,因此type.__new__
可以将它们传递给__init_subclass__
,但是ABCMeta和其他大量的元类都不会那样做。 abc.ABCMeta.__new__
ch x
关键字参数会阻塞而不是传递它,导致您看到的异常。
尝试将__init_subclass__
关键字参数与尚未针对新设计更新的元类使用是行不通的。你必须等待你使用的元类进行修补。