解决元类冲突
- 2025-04-17 09:02:00
- admin 原创
- 16
问题描述:
我需要创建一个根据某些条件使用不同基类的类。对于某些类,我遇到了以下问题:
TypeError: metaclass conflict: the metaclass of a derived class must be a (non-strict) subclass of the metaclasses of all its bases
一个例子是sqlite3
,这里有一个简短的例子,你甚至可以在解释器中使用:
>>> import sqlite3
>>> x = type('x', (sqlite3,), {})
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: metaclass conflict: the metaclass of a derived class must be a (non-strict) subclass of the metaclasses of all its bases
解决方案 1:
您可以直接使用以下方法,而不必使用 jdi 提到的方法:
class M_C(M_A, M_B):
pass
class C(A, B):
__metaclass__ = M_C
解决方案 2:
您使用的示例sqlite3
无效,因为它是一个模块而不是类。我也遇到过这个问题。
您的问题如下:基类的元类与子类的类型不同。这就是为什么您会得到TypeError
。
我使用了此 activestate 代码片段的变体noconflict.py 。由于该代码片段与 Python 3.x 不兼容,因此需要重新编写。无论如何,它应该能给你一个大概的思路。
问题片段
class M_A(type):
pass
class M_B(type):
pass
class A(object):
__metaclass__=M_A
class B(object):
__metaclass__=M_B
class C(A,B):
pass
#Traceback (most recent call last):
# File "<stdin>", line 1, in ?
#TypeError: metaclass conflict: the metaclass of a derived class must be a (non-strict) subclass #of the metaclasses of all its bases
解决方案片段
from noconflict import classmaker
class C(A,B):
__metaclass__=classmaker()
print C
#<class 'C'>
代码配方为您正确地解析了元类。
解决方案 3:
我喜欢做:
class mBase1(type):
...
class mBase2(type):
...
class Base1(metaclass=mBase1):
...
class Base2(metaclass=mBase2):
...
class mChild(type(Base1), type(Base2)):
pass
class Child(Base1, Base2, metaclass=mChild):
...
这样,如果基础的元类发生变化,您就不必担心。type()
会处理好它。
解决方案 4:
当您尝试从函数而不是类继承时也会发生这种情况。
例如。
def function():
pass
class MyClass(function):
pass
解决方案 5:
据我从前面的答案中理解,我们通常需要手动做的唯一事情是:
class M_A(type): pass
class M_B(type): pass
class A(metaclass=M_A): pass
class B(metaclass=M_B): pass
class M_C(M_A, M_B): pass
class C:(A, B, metaclass=M_C): pass
但是我们现在可以通过以下方式自动执行最后两行:
def metaclass_resolver(*classes):
metaclass = tuple(set(type(cls) for cls in classes))
metaclass = metaclass[0] if len(metaclass)==1 \n else type("_".join(mcls.__name__ for mcls in metaclass), metaclass, {}) # class M_C
return metaclass("_".join(cls.__name__ for cls in classes), classes, {}) # class C
class C(metaclass_resolver(A, B)): pass
由于我们不使用任何特定于版本的元类语法,因此它metaclass_resolver
适用于 Python 2 和 Python 3。
解决方案 6:
要使用@michael描述的模式,但要兼容Python 2和3(使用six
库):
from six import with_metaclass
class M_C(M_A, M_B):
pass
class C(with_metaclass(M_C, A, B)):
# implement your class here
相关推荐
热门文章
项目管理软件有哪些?
热门标签
曾咪二维码
扫码咨询,免费领取项目管理大礼包!
云禅道AD