为何类属性与实例属性不一致?
在此处代码里:
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,它定义了 get 和 set 方法。
>>> 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 通过 get 和 set 方法起作用。当 x 是 foo 时,y.x 和 bar.x 通过描述符相互关联。但是,bar.x 被替换为数字 2 后,x 的类型和内存地址发生变化,将 y.x 和 bar.x 分离。
根据官方文档对描述符调用的描述:
- 实例查找优先级:数据描述符 > 实例变量 > 非数据描述符 > 类变量 > __getattr__()
- 找到描述符后,将以 desc.__get__(none, a) 方式调用它,其中 a 是一个类
- 数据描述符始终重写实例字典