web-dev-qa-db-ja.com

メタクラスの競合を解決する

いくつかの条件に応じて、異なる基本クラスを使用するクラスを作成する必要があります。いくつかのクラスで私は悪名高くなります:

TypeError: metaclass conflict: the metaclass of a derived class must be a (non-strict) subclass of the metaclasses of all its bases

1つの例は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
35
Yves Dorfsman

Jdiで言及されているようにレシピを使用する代わりに、直接使用できます:

class M_C(M_A, M_B):
    pass

class C(A, B):
    __metaclass__ = M_C
16
Michael

sqlite3を使用した例は、クラスではなくモジュールであるため無効です。私もこの問題に遭遇しました。

あなたの問題はここにあります:基本クラスには、サブクラスと同じ型ではないメタクラスがあります。それがTypeErrorを取得する理由です。

noconflict.pyを使用したこのactivestateスニペット のバリエーションを使用しました。 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'>

コードレシピは、メタクラスを適切に解決します。

15
jdi

@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
8
Danilo Bargen

これは、クラスではなく関数から継承しようとした場合にも発生します。

例えば。

def function():
    pass

class MyClass(function):
    pass
6
Antwan

前の回答から理解した限り、通常手動で行う必要があると思うのは次のとおりです。

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

ただし、次の方法で最後の2行を自動化できます。

def metaclass_resolver(*classes):
    metaclass = Tuple(set(type(cls) for cls in classes))
    metaclass = metaclass[0] if len(metaclass)==1 \
                else type("_".join(mcls.__for mcls in metaclass), metaclass, {})   # class M_C
    return metaclass("_".join(cls.__for cls in classes), classes, {})              # class C

class C(metaclass_resolver(A, B)): pass

バージョン固有のメタクラス構文を使用しないため、このmetaclass_resolverは、Python 2と同様にPython 3。

4
Chickenmarkus