在python中,我可以使用@classmethod装饰器向类中添加方法。是否有类似的装饰器可以将属性添加到类中?我可以更好地展示我在说什么。
class Example(object):
the_I = 10
def __init__( self ):
self.an_i = 20
@property
def i( self ):
return self.an_i
def inc_i( self ):
self.an_i += 1
# is this even possible?
@classproperty
def I( cls ):
return cls.the_I
@classmethod
def inc_I( cls ):
cls.the_I += 1
e = Example()
assert e.i == 20
e.inc_i()
assert e.i == 21
assert Example.I == 10
Example.inc_I()
assert Example.I == 11
我上面使用的语法是可能的还是需要更多的东西?
我需要类属性的原因是我可以延迟加载类属性,这似乎很合理。
据我所知,如果不创建一个新的元类,就无法为类属性编写setter。
我发现下面的方法是有效的。定义一个具有所需的所有类属性和setter的元类。IE,我需要一个带有setter的title属性的类。以下是我所写的:
class TitleMeta(type):
@property
def title(self):
return getattr(self, '_title', 'Default Title')
@title.setter
def title(self, title):
self._title = title
# Do whatever else you want when the title is set...
现在让您想要的实际类正常,除了让它使用您上面创建的元类。
# Python 2 style:
class ClassWithTitle(object):
__metaclass__ = TitleMeta
# The rest of your class definition...
# Python 3 style:
class ClassWithTitle(object, metaclass = TitleMeta):
# Your class definition...
如果我们只在单个类上使用这个元类,那么像上面那样定义它就有点奇怪了。在这种情况下,如果您使用的是python2风格,您实际上可以在类主体中定义元类。这样它就不在模块范围内定义。
如果您按如下方式定义classproperty,那么您的示例将完全按照您的要求工作。
class classproperty(object):
def __init__(self, f):
self.f = f
def __get__(self, obj, owner):
return self.f(owner)
需要注意的是,您不能将此用于可写属性。While . i = 20将引发AttributeError, Example。I = 20将覆盖属性对象本身。