Python类属性访问的魔法方法:
1. __getattr__(self, name)
- 定义当用户试图获取一个不存在的属性时的行为
2. __getattribute__(self, name)
- 定义当该类的属性被访问时的行为
注意:当__getattr__与__getattribute__同时重写时,访问属性时,优先调用__getattribute__,只有当被访问的属性不存在时才触发__getattr__
3. __setattr__(self, name, value)
- 定义当一个属性被设置时的行为
4. __delattr__(self, name)
- 定义当一个属性被删除时的行为
>>> class C:
def __getattribute__(self, name):
print("getattribute")
return super().__getattribute__(name) #如果这里没有return语句,那么,c.x访问时__getattr__不会触发,因为相当于没有c.x这句话
def __getattr__(self, name):
print("getattr")
def __setattr__(self, name, value):
print("setattr")
super().__setattr__(name, value) #同样的道理,这里必须执行,才能真正的设置成功。这里为啥不能这么写:self.name = value?请看下面关于__setattr__死循环陷阱的说明
def __delattr__(self, name):
print("delattr")
#这里应该加一句 super().__delattr__(name),不然,删除指定对象是不会成功,看下面的执行结果就可以验证 >>> c = C()
>>> c.x
getattribute
getattr
>>> c.x = 999
setattr
>>> c.x
getattribute
999
>>> del c.x
delattr
#上面已经删除了c.x,但下面访问时,还是访问到了,说明删除没有成功,因为__delattr__中没有super().__delattr__(name)这句话
>>> c.x
getattribute
999
>>>
此外,__setattr__会有死循环陷阱:
>>> class Rect():
def __init__(self, width=0, height=0):
self.width = width
self.height = height
def __setattr__(self, name, value):
if name == 'square':
self.width = value
self.height = value
else:
self.name = value
def getArea(self):
return self.width * self.height >>> r = Rect(2,8)
Traceback (most recent call last):
File "<pyshell#144>", line 1, in <module>
r = Rect(2,8)
File "<pyshell#143>", line 3, in __init__
self.width = width
File "<pyshell#143>", line 10, in __setattr__
self.name = value
File "<pyshell#143>", line 10, in __setattr__
self.name = value
File "<pyshell#143>", line 10, in __setattr__
self.name = value
File "<pyshell#143>", line 10, in __setattr__
……
为什么会这样?
主要是在__init__内,给width与height赋值的时候,就会自动触发__setattr__方法,当参数2,8分别传入width与height的时候,初始化时,赋值触发__setattr__,因为2传给的是width属性,所以,不是suqare,就执行else的语句(self.name = value),然而,在else的语句又是一个赋值语句,又会自动触发__setattr__,所以,就会造成死循环。其解决方法有两种:
1.self.name = value这句修改成:super().__setattr__(name, value),使用父类的__setattr__来赋值(这为啥就不会死循环?别问我,反正Python的设计者解决了这个问题)
2.self.name = value这句修改成:self.__dict__[name] = value, 这样也是赋值,为啥不是死循环呢?哈哈,触发__setattr__的情况是访问访问对象的属性,而这里比较巧的是访问的是__dict__(对象的特殊属性,是用来存放当前对象所以的属性的字典)。