8

这是我要解决的一个非常简单的示例:

class Test(object):
    some_dict = {Test: True}

问题是我无法在 Test 仍在定义时引用它

通常,我会这样做:

class Test(object):
    some_dict = {}
    def __init__(self):
        if self.__class__.some_dict == {}:
            self.__class__.some_dict = {Test: True}

但我从不创建此类的实例。它实际上只是一个容纳一组相关函数和数据的容器(我有几个这样的类,并且我传递对它们的引用,因此Test 必须是它自己的类

所以我的问题是,在定义它时我怎么能引用 Test ,或者在定义类后是否有类似于__init__get 的东西?如果可能的话,我想self.some_dict = {Test: True}留在类定义中。到目前为止,这是我知道如何做到这一点的唯一方法:

class Test(object):
    @classmethod
    def class_init(cls):
        cls.some_dict = {Test: True}
Test.class_init()
4

5 回答 5

12

该类在定义时实际上并不存在。语句的工作方式是class语句的主体作为代码块在单独的命名空间中执行。在执行结束时,该命名空间被传递给元类(例如type),并且元类使用命名空间作为属性空间来创建类。

根据您的描述, Test 听起来没有必要成为一个类。听起来它应该是一个模块some_dict是一个全局的——即使它是一个类属性,你的程序中也只有一个这样的属性,所以它并不比拥有一个全局更好——你在类中拥有的任何类方法都可以是函数。

如果你真的想让它成为一个类,你有三个选择:在定义类之后设置字典:

class Test:
    some_dict = {}
Test.some_dict[Test] = True

使用类装饰器(在 Python 2.6 或更高版本中):

def set_some_dict(cls):
    cls.some_dict[cls] = True

@set_some_dict
class Test:
    some_dict = {}

或使用元类:

class SomeDictSetterType(type):
    def __init__(self, name, bases, attrs):
        self.some_dict[self] = True
        super(SomeDictSetterType, self).__init__(name, bases, attrs)

class Test(object):
    __metaclass__ = SomeDictSetterType
    some_dict = {}
于 2010-04-24T22:41:58.640 回答
4

您可以在主类定义之后添加 some_dict 属性。

class Test(object):
  pass
Test.some_dict = {Test: True}
于 2010-04-24T22:38:15.997 回答
1

过去我尝试过以这种方式使用类,但它很快就会变得丑陋(例如,所有方法都需要是类方法或静态方法,你最终可能会意识到你想要定义某些特殊的方法,你必须开始使用元类)。如果您只使用类实例,它可以使事情变得更容易——实际上并没有任何缺点。

其他人建议的(看起来很奇怪的)替代方案:您可以使用__new__

class Test(object):
    def __new__(cls):
        cls.some_dict = {cls: True}

Test()

您甚至可以__new__返回对该类的引用并使用装饰器来调用它:

def instantiate(cls):
    return cls()

@instantiate
class Test(object):
    def __new__(cls):
        cls.some_dict = {cls: True}
        return cls
于 2012-06-07T08:32:12.420 回答
0

您还可以使用元类(这里是一个函数,但还有其他方法):

def Meta(name, bases, ns):
    klass = type(name, bases, ns)
    setattr(klass, 'some_dict', { klass: True })
    return klass

class Test(object):
    __metaclass__ = Meta

print Test.some_dict
于 2010-04-26T07:33:45.407 回答
0

Thomas 的第一个示例非常好,但这里有一种更 Pythonic 的方式来做同样的事情。

class Test:
    x = {}
    @classmethod
    def init(cls):
        # do whatever setup you need here
        cls.x[cls] = True
Test.init()
于 2012-06-07T08:13:26.223 回答