自定义属性访问
__getattr__属性
什么时候触发:
当我们访问属性的时候,如果属性不存在,则触发此方法;
class MyTest: def __getattr__(self, item): # 当访问的属性不存在时,该方法会被触发 print("触发__getattr__属性") super().__getattribute__(item) mt = MyTest() mt.name 执行结果: mt.name File "C:/Users/Administrator/PycharmProjects/untitled/test.py", line 32, in __getattr__ super().__getattribute__(item) AttributeError: 'MyTest' object has no attribute 'name' 触发__getattr__属性
官方文档给出:该方法应当返回一个属性值,或者引发一个异常;
__getattribute__属性
什么时候触发:
查找属性的时候,第一时间触发该方法去查找属性;
class MyTest: def __getattr__(self, item): # 当访问的属性不存在时(报AttrError异常时),该方法会被触发 print("触发__getattr__方法") super().__getattribute__(item) def __getattribute__(self, item): # 当访问属性的时候,第一时间触发此方法去查找属性 print("触发__getattribute__方法") return object.__getattribute__(self,item) mt = MyTest() mt.name 执行结果: 触发__getattribute__方法 Traceback (most recent call last): 触发__getattr__方法 File "C:/Users/Administrator/PycharmProjects/untitled/test.py", line 43, in <module> mt.name File "C:/Users/Administrator/PycharmProjects/untitled/test.py", line 33, in __getattr__ super().__getattribute__(item) AttributeError: 'MyTest' object has no attribute 'name'
通过上面的例子,我们可以看到,当访问属性的时候,首先触发的是__getattribute__方法,由于访问的属性name不存在,于是触发__getattr__方法
__setattr__属性
什么时候触发:
当设置属性的时候触发;
class MyTest: def __setattr__(self, key, value): # 这个方法是在给对象设置属性的时候触发 print("触发__setattr__方法") super().__setattr__(key,value) # key是属性名,value是属性值 mt = MyTest() mt.name = 10 print(mt.name) 执行结果: 触发__setattr__方法 10
可以通过自定义此方法,在设置属性之前做操作
__delattr__属性
class MyTest: def __delattr__(self, item): # 这个方法在删除属性的时候触发 print("触发__delattr__方法") super().__delattr__(item) # item是属性名 mt = MyTest() mt.name = 10 del mt.name # 删除属性 print(mt.name) 执行结果: 触发__delattr__方法 Traceback (most recent call last): File "C:/Users/Administrator/PycharmProjects/untitled/test.py", line 55, in <module> print(mt.name) AttributeError: 'MyTest' object has no attribute 'name'
通过上面的例子,我们可以发现,当删除属性的时候,触发__delattr__方法,我们可以通过定义此方法,在删除属性之前做操作;
****总结:
上面的四个方法都要调用父类的相应方法,是因为,重新定义属性方法后,没有各自对应的能力了,只有调用相应的父类的方法才能到达要求;