本文主要是介绍Python中的类与描述器(Descriptors),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
文章目录
- Python中的类与描述器(Descriptors)
- 描述器的表现
- 描述器
- 非数据描述器
- 数据描述器
- 带`__set__`的数据描述器
- 带`__delete__`的数据描述器
- Python中常见的描述器类
- 练习
Python中的类与描述器(Descriptors)
描述器的表现
魔术方法 | 说明 |
---|---|
__get__(self,instance,owner) | 获取所有者类的属性,定义了该函数,那么该类就是一个"非数据描述器" |
__set__(self,instance,value) | 设置所有者类的属性,如果一个"非数据描述器"定义了该函数,那么就是"数据描述器" |
__delete__(self,instance) | 删除所有者类的属性 |
- 参数介绍:
- self指代当前类的实例,调用者
- instance是个实例,是owner的实例
- owner是属性的所属的类
描述器
- 在类中只要定义了
__get__,__set__,__delete__
三个方法之一,那么该类就是一个描述器- 实现了
__get__
,就是非数据描述器non-data descriptor - 实现了
__get__,__set__
就是数据描述器 data descriptor - 实现了
__get__,__delete__
方法,也是数据描述器 - 实现了
__get__,__set__,__delete
方法,同样也是数据描述器
- 实现了
- 如果一个类的类属性设置为描述器示例,那么它被称为owner属主
- 当该类的类属性被查找、设置、删除时,就会调用描述器相应的方法。
注意:
- 当一个类的类属性被描述器描述那么在该类,或该类的实例访描述器或描述器的属性时,一定会通过
__get__
方法。而该类的类属性的值取决于描述器中__get__
方法的返回值 - 非数据描述器不会改变属性的访问顺序
- 数据描述器会改变属性的访问顺序。会优先访问数据描述器描述的属性。
- 实例化对象在访问父类属性时,继承体系的优先级,高于"数据描述器"修饰的属性的优先级。
- 数据描述器无法改变继承体系中属性访问的优先级,只能改变实例化属性访问的优先级。如果实例化对象访问的属性被父类拦截,其祖先类中被“数据描述器”修饰的属性优先级会被最近父类的属性拦截。
非数据描述器
- 如果类中的类属性被一个非数据描述器描述,那么改类的类属性值取决于非数据描述器中的
__get__
方法的返回值。 - 非数据描述器不会改变属性的访问顺序
- 简单示例:
class A:def __init__(self):self.z = 10def __get__(self, instance, owner):print("--get--被调用",self, instance, owner)return self #如果是None,或其他值,那么b.x或B.x的返回值就是None或其他值。class B:x = A()y = 10z = A()def __init__(self):self.y = 2self.a = A()self.z = 18b = B()
print(b.a)
print(b.y,b.z,b.a)
print("------------------------------")
print(b.a.z) #不会调用A中的__get__
print(b.__class__.z) #可以看到instance值为None,因为不是B的实例调用
print("***********")
print(b.x.z) #会调用A中的__get__
print(B.x.z) #会调用A中的__get__
print("***********")
B.x = 10
print(b.x)
- 上面示例可以看出:
__get__(self,instance,owner)
方法的签名,会传入3个参数- B.x调用时,self表示A的实例,instance为None,owner为B类
- b.x调用时,self表示A的实例,instance为B的实例b,owner为B类
数据描述器
带__set__
的数据描述器
-
实现了
__get__,__set__
就是数据描述器 data descriptor -
数据描述器会改变属性的访问顺序。会优先访问数据描述器描述的属性。
-
3.6新增描述器方法
__set_name__(self,owner,name)
,它在属主类构建的时候会调用。(注意:目前动态的为属主owner类使用setattr注入类属性时,不会调用(本人怀疑可以能是bug)) -
实现了
__get__,__delete__
方法,也是数据描述器 -
简单示例:
class A:def __init__(self):self.z = 10def __get__(self, instance, owner):prin
这篇关于Python中的类与描述器(Descriptors)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!