Hello! 欢迎来到小浪资源网!

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


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

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

在此处代码里:

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 是一个类
  • 数据描述符始终重写实例字典

相关阅读