Наследование методов из метакласса - PullRequest
2 голосов
/ 01 декабря 2009

В приведенном ниже примере кода перечисления в этот вопрос , воспроизведенный ниже, почему TOKEN содержит реализации __contains__ и __repr__ из метакласса EnumerationType?

from ctypes import *

class EnumerationType(type(c_uint)):  
    def __new__(metacls, name, bases, dict):  
        if not "_members_" in dict:  
            _members_ = {}  
            for key,value in dict.items():  
                if not key.startswith("_"):  
                    _members_[key] = value  
            dict["_members_"] = _members_  
        cls = type(c_uint).__new__(metacls, name, bases, dict)  
        for key,value in cls._members_.items():  
            globals()[key] = value  
        return cls  

    def __contains__(self, value):
        return value in self._members_.values()

    def __repr__(self):
        return "<Enumeration %s>" % self.__name__

class Enumeration(c_uint):
    __metaclass__ = EnumerationType
    _members_ = {}
    def __init__(self, value):
        for k,v in self._members_.items():
            if v == value:
                self.name = k
                break
        else:
            raise ValueError("No enumeration member with value %r" % value)
        c_uint.__init__(self, value)


    @classmethod
    def from_param(cls, param):
        if isinstance(param, Enumeration):
            if param.__class__ != cls:
                raise ValueError("Cannot mix enumeration members")
            else:
                return param
        else:
            return cls(param)

    def __repr__(self):
        return "<member %s=%d of %r>" % (self.name, self.value, self.__class__)

class TOKEN(Enumeration):
    _members_ = {'T_UNDEF':0, 'T_NAME':1, 'T_NUMBER':2, 'T_STRING':3, 'T_OPERATOR':4, 'T_VARIABLE':5, 'T_FUNCTION':6}

Я ожидаю, что в следующем коде будут сгенерированы исключения для того, чтобы __contains__ не был реализован, но вместо этого я получил True False.

print 2 in TOKEN, 7 in TOKEN

1 Ответ

2 голосов
/ 01 декабря 2009

И Enumeration, и TOKEN являются экземплярами EnumerationType:

>>> isinstance(Enumeration, EnumerationType)
True
>>> isinstance(TOKEN, EnumerationType)
True

И специальные методы для экземпляров новых классов стилей ищутся в классе , например repr(TOKEN) эквивалентно type(TOKEN).__repr__(TOKEN), то есть EnumerationType.__repr__(TOKEN).

...