Использование свойства () для методов класса - PullRequest
148 голосов
/ 24 сентября 2008

У меня есть класс с двумя методами класса (использующими функцию classmethod ()) для получения и установки того, что по сути является статической переменной. Я пытался использовать функцию property () с ними, но это приводит к ошибке. Мне удалось воспроизвести ошибку со следующим в интерпретаторе:

class Foo(object):
    _var = 5
    @classmethod
    def getvar(cls):
        return cls._var
    @classmethod
    def setvar(cls, value):
        cls._var = value
    var = property(getvar, setvar)

Я могу продемонстрировать методы класса, но они не работают как свойства:

>>> f = Foo()
>>> f.getvar()
5
>>> f.setvar(4)
>>> f.getvar()
4
>>> f.var
Traceback (most recent call last):
  File "<stdin>", line 1, in ?
TypeError: 'classmethod' object is not callable
>>> f.var=5
Traceback (most recent call last):
  File "<stdin>", line 1, in ?
TypeError: 'classmethod' object is not callable

Можно ли использовать функцию property () для функций, декорированных методом класса?

Ответы [ 13 ]

70 голосов
/ 26 ноября 2009

Свойство создается в классе, но влияет на экземпляр. Поэтому, если вам нужно свойство classmethod, создайте свойство в метаклассе.

>>> class foo(object):
...     _var = 5
...     class __metaclass__(type):  # Python 2 syntax for metaclasses
...         pass
...     @classmethod
...     def getvar(cls):
...         return cls._var
...     @classmethod
...     def setvar(cls, value):
...         cls._var = value
...     
>>> foo.__metaclass__.var = property(foo.getvar.im_func, foo.setvar.im_func)
>>> foo.var
5
>>> foo.var = 3
>>> foo.var
3

Но так как вы в любом случае используете метакласс, будет лучше читать, если вы просто переместите туда методы класса.

>>> class foo(object):
...     _var = 5
...     class __metaclass__(type):  # Python 2 syntax for metaclasses
...         @property
...         def var(cls):
...             return cls._var
...         @var.setter
...         def var(cls, value):
...             cls._var = value
... 
>>> foo.var
5
>>> foo.var = 3
>>> foo.var
3

или с использованием синтаксиса metaclass=... в Python 3, метакласса, определенного вне тела класса foo, и метакласса, ответственного за установку начального значения _var:

>>> class foo_meta(type):
...     def __init__(cls, *args, **kwargs):
...         cls._var = 5
...     @property
...     def var(cls):
...         return cls._var
...     @var.setter
...     def var(cls, value):
...         cls._var = value
...
>>> class foo(metaclass=foo_meta):
...     pass
...
>>> foo.var
5
>>> foo.var = 3
>>> foo.var
3
69 голосов
/ 05 сентября 2009

Читая заметки Python 2.2 , я нахожу следующее.

Метод get [свойства] не будет вызываться, когда свойство доступно как класс атрибут (C.x), а не как атрибут экземпляра (C (). x). если ты хотите переопределить операцию __get__ для свойств при использовании в качестве класса атрибут, вы можете наследовать свойство - это новый тип стиля сам по себе расширить его метод __get__, или вы можете определить тип дескриптора с нуля создав новый класс в стиле определяет __get__, __set__ и __delete__ методы.

ПРИМЕЧАНИЕ. Приведенный ниже метод фактически не работает для сеттеров, а только для геттеров.

Поэтому я считаю, что предписанное решение заключается в создании ClassProperty в качестве подкласса свойства.

class ClassProperty(property):
    def __get__(self, cls, owner):
        return self.fget.__get__(None, owner)()

class foo(object):
    _var=5
    def getvar(cls):
        return cls._var
    getvar=classmethod(getvar)
    def setvar(cls,value):
        cls._var=value
    setvar=classmethod(setvar)
    var=ClassProperty(getvar,setvar)

assert foo.getvar() == 5
foo.setvar(4)
assert foo.getvar() == 4
assert foo.var == 4
foo.var = 3
assert foo.var == 3

Однако, на самом деле сеттеры не работают:

foo.var = 4
assert foo.var == foo._var # raises AssertionError

foo._var без изменений, вы просто перезаписали свойство новым значением.

Вы также можете использовать ClassProperty в качестве декоратора:

class foo(object):
    _var = 5

    @ClassProperty
    @classmethod
    def var(cls):
        return cls._var

    @var.setter
    @classmethod
    def var(cls, value):
        cls._var = value

assert foo.var == 5
44 голосов
/ 29 ноября 2012

Надеюсь, этот простой-простой для чтения @classproperty декоратор поможет кому-то искать свойства класса.

class classproperty(object):

    def __init__(self, fget):
        self.fget = fget

    def __get__(self, owner_self, owner_cls):
        return self.fget(owner_cls)

class C(object):

    @classproperty
    def x(cls):
        return 1

assert C.x == 1
assert C().x == 1
21 голосов
/ 17 сентября 2016

Можно ли использовать функцию property () для функций, декорированных методом класса?

Нет.

Однако метод класса - это просто связанный метод (частичная функция) класса, доступный из экземпляров этого класса.

Поскольку экземпляр является функцией класса, и вы можете извлечь класс из экземпляра, вы можете получить любое желаемое поведение из свойства класса с помощью property:

class Example(object):
    _class_property = None
    @property
    def class_property(self):
        return self._class_property
    @class_property.setter
    def class_property(self, value):
        type(self)._class_property = value
    @class_property.deleter
    def class_property(self):
        del type(self)._class_property

Этот код можно использовать для тестирования - он должен пройти без ошибок:

ex1 = Example()
ex2 = Example()
ex1.class_property = None
ex2.class_property = 'Example'
assert ex1.class_property is ex2.class_property
del ex2.class_property
assert not hasattr(ex1, 'class_property')

И обратите внимание, что нам вообще не нужны метаклассы - и вы в любом случае напрямую не обращаетесь к метаклассу через экземпляры его классов.

написание @classproperty декоратора

Вы можете создать декоратор classproperty всего за несколько строк кода, создав подкласс property (он реализован в C, но вы можете увидеть эквивалентный Python здесь ):

class classproperty(property):
    def __get__(self, obj, objtype=None):
        return super(classproperty, self).__get__(objtype)
    def __set__(self, obj, value):
        super(classproperty, self).__set__(type(obj), value)
    def __delete__(self, obj):
        super(classproperty, self).__delete__(type(obj))

Затем обработайте декоратор, как если бы это был метод класса, объединенный со свойством:

class Foo(object):
    _bar = 5
    @classproperty
    def bar(cls):
        """this is the bar attribute - each subclass of Foo gets its own.
        Lookups should follow the method resolution order.
        """
        return cls._bar
    @bar.setter
    def bar(cls, value):
        cls._bar = value
    @bar.deleter
    def bar(cls):
        del cls._bar

И этот код должен работать без ошибок:

def main():
    f = Foo()
    print(f.bar)
    f.bar = 4
    print(f.bar)
    del f.bar
    try:
        f.bar
    except AttributeError:
        pass
    else:
        raise RuntimeError('f.bar must have worked - inconceivable!')
    help(f)  # includes the Foo.bar help.
    f.bar = 5

    class Bar(Foo):
        "a subclass of Foo, nothing more"
    help(Bar) # includes the Foo.bar help!
    b = Bar()
    b.bar = 'baz'
    print(b.bar) # prints baz
    del b.bar
    print(b.bar) # prints 5 - looked up from Foo!


if __name__ == '__main__':
    main()

Но я не уверен, насколько это было бы хорошо. Старый список рассылки статья предполагает, что это не должно работать.

Получение имущества для работы в классе:

Недостатком вышеизложенного является то, что «свойство класса» недоступно из класса, поскольку оно просто перезапишет дескриптор данных из класса __dict__.

Однако мы можем переопределить это свойство, определенное в метаклассе __dict__. Например:

class MetaWithFooClassProperty(type):
    @property
    def foo(cls):
        """The foo property is a function of the class -
        in this case, the trivial case of the identity function.
        """
        return cls

И тогда экземпляр класса метакласса может иметь свойство, которое обращается к свойству класса, используя принцип, уже продемонстрированный в предыдущих разделах:

class FooClassProperty(metaclass=MetaWithFooClassProperty):
    @property
    def foo(self):
        """access the class's property"""
        return type(self).foo

И теперь мы видим оба экземпляра

>>> FooClassProperty().foo
<class '__main__.FooClassProperty'>

и класс

>>> FooClassProperty.foo
<class '__main__.FooClassProperty'>

есть доступ к свойству класса.

17 голосов
/ 16 ноября 2017

Python 3!

Старый вопрос, множество мнений, крайне нуждающихся в единственно верном Python 3 пути.

К счастью, с metaclass kwarg это просто:

class FooProperties(type):

    @property
    def var(cls):
        return cls._var

class Foo(object, metaclass=FooProperties):
    _var = 'FOO!'

Тогда >>> Foo.var

'Foo!

15 голосов
/ 25 сентября 2008

Нет разумного способа заставить эту систему "свойства класса" работать в Python.

Вот один из необоснованных способов заставить его работать. Вы, безусловно, можете сделать его более плавным с увеличением количества магии метаклассов.

class ClassProperty(object):
    def __init__(self, getter, setter):
        self.getter = getter
        self.setter = setter
    def __get__(self, cls, owner):
        return getattr(cls, self.getter)()
    def __set__(self, cls, value):
        getattr(cls, self.setter)(value)

class MetaFoo(type):
    var = ClassProperty('getvar', 'setvar')

class Foo(object):
    __metaclass__ = MetaFoo
    _var = 5
    @classmethod
    def getvar(cls):
        print "Getting var =", cls._var
        return cls._var
    @classmethod
    def setvar(cls, value):
        print "Setting var =", value
        cls._var = value

x = Foo.var
print "Foo.var = ", x
Foo.var = 42
x = Foo.var
print "Foo.var = ", x

Узел проблемы в том, что свойства - это то, что Python называет "дескрипторами". Нет короткого и простого способа объяснить, как работает этот тип метапрограммирования, поэтому я должен указать вам на дескриптор howto .

Вам нужно когда-либо разбираться в подобных вещах, только если вы внедряете довольно продвинутую среду. Например, система прозрачности объектов, система RPC или разновидность языка, специфичного для предметной области.

Однако в комментарии к предыдущему ответу вы говорите, что вы

необходимо изменить атрибут, который таким образом, который виден всеми экземплярами класса, и в области действия, из которой вызываются эти методы класса, не имеет ссылок на все экземпляры класса.

Мне кажется, что вам действительно нужен шаблон проектирования Observer .

5 голосов
/ 30 марта 2010

Установка его только для мета-класса не поможет, если вы хотите получить доступ к свойству класса через инстанцированный объект, в этом случае вам также необходимо установить обычное свойство для объекта (который отправляет свойство класса) , Я думаю, что следующее немного яснее:

#!/usr/bin/python

class classproperty(property):
    def __get__(self, obj, type_):
        return self.fget.__get__(None, type_)()

    def __set__(self, obj, value):
        cls = type(obj)
        return self.fset.__get__(None, cls)(value)

class A (object):

    _foo = 1

    @classproperty
    @classmethod
    def foo(cls):
        return cls._foo

    @foo.setter
    @classmethod
    def foo(cls, value):
        cls.foo = value

a = A()

print a.foo

b = A()

print b.foo

b.foo = 5

print a.foo

A.foo = 10

print b.foo

print A.foo
2 голосов
/ 25 сентября 2008

Поскольку мне нужно изменить атрибут, который таким образом, который виден всем экземплярам класса, и в области, из которой вызываются эти методы класса, не имеет ссылок на все экземпляры класса.

У вас есть доступ хотя бы к одному экземпляру класса? Тогда я могу придумать, как это сделать:

class MyClass (object):
    __var = None

    def _set_var (self, value):
        type (self).__var = value

    def _get_var (self):
        return self.__var

    var = property (_get_var, _set_var)

a = MyClass ()
b = MyClass ()
a.var = "foo"
print b.var
2 голосов
/ 24 сентября 2008

Половина решения, __set__ в классе не работает, по-прежнему. Решением является пользовательский класс свойств, реализующий как свойство, так и статический метод

class ClassProperty(object):
    def __init__(self, fget, fset):
        self.fget = fget
        self.fset = fset

    def __get__(self, instance, owner):
        return self.fget()

    def __set__(self, instance, value):
        self.fset(value)

class Foo(object):
    _bar = 1
    def get_bar():
        print 'getting'
        return Foo._bar

    def set_bar(value):
        print 'setting'
        Foo._bar = value

    bar = ClassProperty(get_bar, set_bar)

f = Foo()
#__get__ works
f.bar
Foo.bar

f.bar = 2
Foo.bar = 3 #__set__ does not
1 голос
/ 28 июня 2017

Вот решение, которое должно работать как для доступа через класс, так и для доступа через экземпляр, использующий метакласс.

In [1]: class ClassPropertyMeta(type):
   ...:     @property
   ...:     def prop(cls):
   ...:         return cls._prop
   ...:     def __new__(cls, name, parents, dct):
   ...:         # This makes overriding __getattr__ and __setattr__ in the class impossible, but should be fixable
   ...:         dct['__getattr__'] = classmethod(lambda cls, attr: getattr(cls, attr))
   ...:         dct['__setattr__'] = classmethod(lambda cls, attr, val: setattr(cls, attr, val))
   ...:         return super(ClassPropertyMeta, cls).__new__(cls, name, parents, dct)
   ...:

In [2]: class ClassProperty(object):
   ...:     __metaclass__ = ClassPropertyMeta
   ...:     _prop = 42
   ...:     def __getattr__(self, attr):
   ...:         raise Exception('Never gets called')
   ...:

In [3]: ClassProperty.prop
Out[3]: 42

In [4]: ClassProperty.prop = 1
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-4-e2e8b423818a> in <module>()
----> 1 ClassProperty.prop = 1

AttributeError: can't set attribute

In [5]: cp = ClassProperty()

In [6]: cp.prop
Out[6]: 42

In [7]: cp.prop = 1
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-7-e8284a3ee950> in <module>()
----> 1 cp.prop = 1

<ipython-input-1-16b7c320d521> in <lambda>(cls, attr, val)
      6         # This makes overriding __getattr__ and __setattr__ in the class impossible, but should be fixable
      7         dct['__getattr__'] = classmethod(lambda cls, attr: getattr(cls, attr))
----> 8         dct['__setattr__'] = classmethod(lambda cls, attr, val: setattr(cls, attr, val))
      9         return super(ClassPropertyMeta, cls).__new__(cls, name, parents, dct)

AttributeError: can't set attribute

Это также работает с сеттером, определенным в метаклассе.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...