>백엔드 개발 >파이썬 튜토리얼 >Python에서 클래스 메서드와 함께 `property()`를 사용할 수 없는 이유와 해결 방법은 무엇입니까?

Python에서 클래스 메서드와 함께 `property()`를 사용할 수 없는 이유와 해결 방법은 무엇입니까?

DDD
DDD원래의
2024-12-12 21:28:12267검색

Why Can't I Use `property()` with Classmethods in Python, and What Are the Workarounds?

클래스 메소드와 함께 property() 사용: 탐색

Python에서 property() 함수는 getter 및 setter를 생성하는 데 중요한 역할을 합니다. 클래스 속성의 경우. 그러나 @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)

Foo 클래스의 인스턴스에서 var 속성에 액세스하거나 설정하려고 하면 다음과 같은 오류가 발생합니다. 오류:

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

문서 및 스택 오버플로 토론에 설명된 대로 [Setter 및 Getter로서의 Python 클래스 메서드](https://stackoverflow.com/questions/27906399/python-class-method-as -setter-and-getter), Python 3.8에서는 클래스 메서드와 함께 property() 함수를 사용할 수 없습니다.

그러나 원하는 기능을 달성하기 위한 대체 접근 방식이 있습니다.

Python 2 및 Python 3(3.9-3.10 포함):

속성은 인스턴스에 영향을 주지만 클래스 속성은 클래스에 생성되므로 인스턴스에 속성을 생성해야 합니다. Metaclass.

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)

3.9 이상의 Python 버전:

더 깔끔한 솔루션은 클래스 메서드를 메타클래스로 이동하고 메타클래스에서 직접 속성을 선언하는 것입니다.

class Foo_meta(type):
    @property
    def var(cls):
        return cls._var

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

class Foo(metaclass=Foo_meta):
    _var = 5

이러한 대체 방법을 사용하면 클래스 수준에서 효과적으로 사용할 수 있습니다. 속성.

위 내용은 Python에서 클래스 메서드와 함께 `property()`를 사용할 수 없는 이유와 해결 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.