首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >子类ctypes Python

子类ctypes Python
EN

Stack Overflow用户
提问于 2009-05-13 03:36:33
回答 2查看 1.1K关注 0票数 4

这是我在网上找到的一些代码。我不知道它是怎么用的。我简单地用枚举键/值填充成员,它可以工作,但是我很好奇这个元类到底是关于什么的。我假设它与ctype有关,但是我找不到关于子类ctype的很多信息。我知道EnumerationType并没有像我使用枚举那样做任何事情。

代码语言:javascript
复制
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}
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2009-05-13 03:41:27

元类是用来创建类的类。可以这样想:所有的对象都有一个类,一个类也是一个对象,因此,类可以有一个类是有意义的。

http://www.ibm.com/developerworks/linux/library/l-pymeta.html

要理解这是在做什么,您可以查看代码中的几个要点。

代码语言:javascript
复制
 _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”,并使它们在您的全局词典中可用。

代码语言:javascript
复制
def __init__(self, value):
    for k,v in self._members_.items():
        if v == value:
            self.name = k
            break

每当您创建枚举实例时,当初始化类时,它将检查“值”是否在允许枚举名称列表中。当找到该值时,它将字符串名称设置为self.name。

代码语言:javascript
复制
c_uint.__init__(self, value)

这是将“ctype值”设置为实际c++无符号整数的实际行。

票数 4
EN

Stack Overflow用户

发布于 2009-05-13 07:47:21

那确实是一门奇怪的课。

您使用它的方式是正确的,尽管另一种方法是:

代码语言:javascript
复制
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行的用途)

然后,您可以这样使用它:

代码语言:javascript
复制
>>> 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对象的方法。不太确定这是否真的是它的目的。

我认为这个类是用来处理使用c风格枚举的外部API时使用的,但是它看起来是一个很大的工作量,却没有什么收获。

票数 3
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/855941

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档