Python类属性访问的魔法方法

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__(对象的特殊属性,是用来存放当前对象所以的属性的字典)。

posted @ 2016-01-14 23:12  翌逍  阅读(1028)  评论(0编辑  收藏  举报