Python 描述符


描述符(Descriptor)是 Python 中一个非常强大的特性,它允许我们自定义属性的访问行为。使用描述符,我们可以创建一些特殊的属性,在访问这些属性时执行自定义的逻辑,如数据验证、属性计算等。

类型:

**数据描述符:**用于修改属性的访问和修改行为。
**方法描述符:**用于修改方法的行为。

数据描述符:

**get:**在属性被访问时被调用。
**set:**在属性被设置时被调用。
**delete:**在属性被删除时被调用。

方法描述符:

**call:**在方法被调用时被调用。

下面我们来看一个具体的例子:

class MyDescriptor:
    def __init__(self, initial_value=None):
        self._value = initial_value

    def __get__(self, instance, owner):
        print(f"Getting value: {self._value}")
        return self._value

    def __set__(self, instance, value):
        print(f"Setting value to: {value}")
        self._value = value

    def __delete__(self, instance):
        print("Deleting value")
        del self._value

class MyClass:
    my_attr = MyDescriptor(42)

obj = MyClass()
print(obj.my_attr)  # Output: Getting value: 42
obj.my_attr = 100   # Output: Setting value to: 100
del obj.my_attr     # Output: Deleting value

在这个例子中,我们定义了一个 MyDescriptor 类,它实现了三个描述符协议方法:__get____set____delete__。这些方法分别在访问、设置和删除属性时被调用。

MyClass 中,我们定义了一个 my_attr 属性,它使用 MyDescriptor 作为描述符。当我们访问、设置或删除 obj.my_attr 时,相应的描述符方法会被调用,并执行我们自定义的逻辑。

描述符的要包括以下几点:

  1. 数据验证: 可以在 __set__ 方法中添加数据验证逻辑,确保属性值符合预期。
  2. 属性计算: 在 __get__ 方法中实现动态计算属性值的逻辑。
  3. 属性缓存: 使用描述符可以实现属性值的缓存,提高访问性能。
  4. 懒加载: 描述符可以用于实现懒加载,即在第一次访问属性时才计算或加载属性值。
  5. 属性权限控制: 可以使用描述符实现只读、只写或读写属性。
  6. 属性依赖管理: 描述符可以用于管理属性之间的依赖关系,确保属性值的一致性。

下面是一个的代码示例,实现了一个带有数据验证和属性缓存的描述符:

class CachedProperty:
    def __init__(self, getter):
        self.getter = getter
        self._cache = {}

    def __get__(self, instance, owner):
        if instance is None:
            return self
        if instance not in self._cache:
            self._cache[instance] = self.getter(instance)
        return self._cache[instance]

    def __set__(self, instance, value):
        self._cache[instance] = value

    def __delete__(self, instance):
        if instance in self._cache:
            del self._cache[instance]

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @CachedProperty
    def full_name(self):
        print("Calculating full name...")
        return f"{self.name} Smith"

    @property
    def age(self):
        return self._age

    @age.setter
    def age(self, value):
        if value < 0:
            raise ValueError("Age cannot be negative")
        self._age = value

p = Person("John", 30)
print(p.full_name)  # Output: Calculating full name... John Smith
print(p.full_name)  # Output: John Smith (from cache)

p.age = 40
print(p.age)  # Output: 40

p.age = -10  # Raises ValueError: Age cannot be negative

在这个场景下,selfinstance 的区别如下:

  1. self:

    • self 代表的是 Person 类本身的实例,也就是 Person 类的一个对象。
    • __get__ 方法中,self 指向的是 Person 类的 age 属性本身,而不是某个特定的 Person 对象。
  2. instance:

    • instance 代表的是正在访问 age 属性的 Person 对象实例。
    • __get__ 方法中,instance 指向的是调用 age 属性的具体 Person 对象,比如上例中的 person 对象。

简单来说:

  • self 指向的是属性本身(即 age 属性),而 instance 指向的是正在访问该属性的对象实例。
  • self 是属性级别的,而 instance 是对象级别的。
  • owner 是属性的类对象

这个区别很重要,因为在 __get__ 方法中,我们需要根据具体的 Person 对象实例(instance)来计算年龄,而不是直接使用 self(即 age 属性本身)。

方法描述符

class Calculator:
    def __init__(self):
        self.num1 = 0
        self.num2 = 0

    def __call__(self, a, b):
        self.num1 = a
        self.num2 = b
        return self

    def add(self):
        return self.num1 + self.num2


calc = Calculator()
result = calc(10, 20)
print(result)  # 结果为30

解释:

  • __call__ 方法描述符允许将 Calculator 类本身作为函数调用。
  • __call__ 方法中,self 参数表示 Calculator 对象,ab 参数表示方法参数。
  • 方法返回 Calculator 对象本身,以便可以继续使用其方法。

优点:

  • 简化了方法调用过程。
  • 允许将类作为函数使用。
  • 提高了代码可读性。

注意事项:

  • __call__ 方法描述符仅适用于类。
  • 如果 __call__ 方法描述符不正确定义,会导致错误。

实现缓存


在这个例子中,当我们尝试访问一个实例对象的属性时,__get__ 方法会被调用。如果实例对象没有该属性的缓存值,它会调用 self.func(instance) 来计算属性值,并将其缓存在实例对象上。

这种技术被称为"惰性计算"(lazy evaluation),它可以提高性能,因为属性值只有在第一次被访问时才会计算。之后,后续的访问都会直接返回缓存的值,而不需要再次计算。

下面是一个更具体的例子:

class LazyProperty:
    def __init__(self, func):
        self.func = func
        self.cache_name = f"_{func.__name__}"

    def __get__(self, instance, owner):
        if instance is None:
            return self
        if not hasattr(instance, self.cache_name):
            value = self.func(instance)
            setattr(instance, self.cache_name, value)
        return getattr(instance, self.cache_name)

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @LazyProperty
    def full_name(self):
        print("Calculating full name...")
        return f"{self.name} Smith"

person = Person("Alice", 30)
print(person.full_name)  # 输出: "Calculating full name..." 和 "Alice Smith"
print(person.full_name)  # 输出: "Alice Smith"

在这个例子中,我们定义了一个 LazyProperty 描述符类,它在第一次访问 full_name 属性时计算并缓存该值。后续访问都会直接返回缓存的值,而不需要再次计算。

总的来说,self.func(instance) 是描述符对象用来计算属性值的方法调用。通过使用描述符,我们可以自定义属性的访问行为,实现惰性计算等优化手段,提高代码的性能和可维护性。

相关推荐

  1. Python 描述符

    2024-04-08 23:56:02       15 阅读
  2. Python高级用法:描述符(descriptor)

    2024-04-08 23:56:02       32 阅读
  3. Python学习之-描述符详解

    2024-04-08 23:56:02       19 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-04-08 23:56:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-04-08 23:56:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-04-08 23:56:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-04-08 23:56:02       20 阅读

热门阅读

  1. 常用启发式算法简介:从迷宫到机器学习

    2024-04-08 23:56:02       15 阅读
  2. SPMI 协议简介

    2024-04-08 23:56:02       14 阅读
  3. C++递推算法

    2024-04-08 23:56:02       15 阅读
  4. 网络通信的隐形护卫

    2024-04-08 23:56:02       14 阅读
  5. 10.左右相同(省模拟赛)

    2024-04-08 23:56:02       14 阅读
  6. python 函数

    2024-04-08 23:56:02       13 阅读
  7. [RK-Linux] RK3399启动流程详解

    2024-04-08 23:56:02       16 阅读
  8. 数据库的介绍、分类、作用和特点

    2024-04-08 23:56:02       15 阅读
  9. 【Go高阶】细说 Channel 的进阶用法

    2024-04-08 23:56:02       15 阅读
  10. Docker Desktop安装

    2024-04-08 23:56:02       16 阅读
  11. react native 相机拍照

    2024-04-08 23:56:02       14 阅读
  12. 贪婪算法python实现

    2024-04-08 23:56:02       18 阅读
  13. nuxt3使用记录二:页面构建的细节(特别是SSG)

    2024-04-08 23:56:02       15 阅读
  14. es6新增加的语法

    2024-04-08 23:56:02       14 阅读