这是我在互联网上找到的一些代码。我不确定它是如何被使用的。我只是使用枚举键/值填充成员并且它有效,但我很好奇这个元类是什么。我假设它与ctypes有关,但我找不到有关子类化ctypes的更多信息。我知道EnumerationType没有像我使用Enumeration那样做任何事情。
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__)
And an enumeration probably done the wrong way.
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}
答案 0 :(得分:4)
元类是用于创建类的类。可以这样想:所有对象都有一个类,一个类也是一个对象,因此,一个类可以有一个类是有意义的。
http://www.ibm.com/developerworks/linux/library/l-pymeta.html
要了解这是做什么的,您可以查看代码中的几点。
_members_ = {'T_UNDEF':0, 'T_NAME':1, 'T_NUMBER':2, 'T_STRING':3, 'T_OPERATOR':4, 'T_VARIABLE':5, 'T_FUNCTION':6}
globals()[key] = value
这里它需要你字典中的每个定义的键:“T_UNDEF”“T_NUMBER”并使它们在你的全局字典中可用。
def __init__(self, value):
for k,v in self._members_.items():
if v == value:
self.name = k
break
每当你创建枚举的实例时,它会在你初始化类时检查“值”是否在你允许的枚举名列表中。找到该值后,它会将字符串名称设置为self.name。
c_uint.__init__(self, value)
这是将“ctypes值”设置为实际c无符号整数的实际行。
答案 1 :(得分:3)
这确实是一个奇怪的阶级。
您使用它的方式是正确的,但另一种方式是:
class TOKEN(Enumeration):
T_UNDEF = 0
T_NAME = 1
T_NUMBER = 2
T_STRING = 3
T_OPERATOR = 4
T_VARIABLE = 5
T_FUNCTION = 6
(这就是__new__
中的前6行)
然后你可以像这样使用它:
>>> TOKEN
<Enumeration TOKEN>
>>> TOKEN(T_NAME)
<member T_NAME=1 of <Enumeration TOKEN>>
>>> T_NAME in TOKEN
True
>>> TOKEN(1).name
'T_NAME'
from_param
方法似乎是为了方便,编写接受int或Enumeration
对象的方法。不确定这是否真的是它的目的。
我认为这个类在使用外部API时可以使用c风格的枚举,但它看起来像是一大堆工作而收效甚微。