首页 > 代码库 > python __setattr__, __getattr__, __delattr__,__getattribute__

python __setattr__, __getattr__, __delattr__,__getattribute__

参考资料

  

  

  

__setattr__、__getattr__和__delattr__以及__getattribute__可以拦截对对象属性的访问;

  

  

>>> s = Something()

>>> s.age = 3

set ‘age‘ = 3

>>> s.age

3

注意到,s.age并没有调用__getattr__,是因为__getattr__方法只在属性没有找到的时候调用

  

虽然相比于Property,实现优点复杂(可能效率更低?),但是足够灵活,功能强大;

  

另外,对于上述代码,访问未绑定的属性,将引发异常

>>> s.name

get ‘name‘

Traceback (most recent call last):

File "<stdin>", line 1, in <module>

File "<stdin>", line 4, in __getattr__

KeyError: ‘name‘

注意:因为name没有绑定到s,所以__getattr__被调用,而用不存在的key访问dictionary会引发KeyError异常;

  

更严重的是,改变了hasattr的行为

>>> hasattr(s,‘name‘)

get ‘name‘

Traceback (most recent call last):

File "<stdin>", line 1, in <module>

File "<stdin>", line 4, in __getattr__

KeyError: ‘name‘

  

  

需要注意的是,防止循环调用,所以__dict__被用用来代替普通的特性赋值操作;另外,__getattribute__拦截所有属性的访问(在新式类中),也拦截对__dict__的访问!访问__getattribute__中与self相关的属性时,使用超类的__getattribute__方法(使用super函数)是唯一安全的途径;

  

>>> s = Something()

>>> s.name

get name

Traceback (most recent call last):

File "<stdin>", line 1, in <module>

File "<stdin>", line 4, in __getattribute__

AttributeError: ‘Something‘ object has no attribute ‘name‘

>>> s.name = 3

>>> s.name

get name

3

  

__getattribute__每次都被调用,不管属性是否被绑定;

  

__getattribute__没有__getattr__的异常问题:

>>> hasattr(s,‘t‘)

False

  

使用__getattribute__的时候,要小心,因为它拦截所有的属性访问,包括__dict__,所以

def __getattribute__(self, key):

    …

    return self.__dict__[key]

会发生循环调用,使用super对象调用__getattribute__可以避免;

  

  

python __setattr__, __getattr__, __delattr__,__getattribute__