类属性和实例属性为何在使用描述符后出现不一致?

类属性和实例属性为何在使用描述符后出现不一致?

为何类属性与实例属性不一致?

在此处代码里:

class foo:
    def __get__(self, instance, owner):
        pass
    def __set__(self, instance, value):
        pass


class bar:
    x = foo()
    def __init__(self, n):
        self.x = n

y 和 bar 实例的初始值相等,因为双方都使用描述符 foo,它定义了 getset 方法。

>>> y = bar(1)
>>> y.x
>>> bar.x
>>> bar.x == y.x
true

但是,当将 bar.x 更改为常规数据类型 2 时,y 和 bar 的属性分离。

>>> Bar.x = 2
>>> y.x
2
>>> Bar.x
2
>>> y = Bar(3)
>>> y.x
3
>>> Bar.x
2

这是因为描述符 foo 通过 getset 方法起作用。当 x 是 foo 时,y.x 和 bar.x 通过描述符相互关联。但是,bar.x 被替换为数字 2 后,x 的类型和内存地址发生变化,将 y.x 和 bar.x 分离。

根据官方文档对描述符调用的描述:

  • 实例查找优先级:数据描述符 > 实例变量 > 非数据描述符 > 类变量 > __getattr__()
  • 找到描述符后,将以 desc.__get__(none, a) 方式调用它,其中 a 是一个类
  • 数据描述符始终重写实例字典

以上就是类属性和实例属性为何在使用描述符后出现不一致?的详细内容,更多请关注硕下网其它相关文章!