先介绍几个类中的应用__getattr__,__setattr__,__get__,__set__,__getattribute__,。
__getattr__:当在类中找不到attribute的时候,会调用__getattr__,并执行其中的自定义代码。所有在类中定义的属性都包含在__dict__中,也就是说如果在__dict__中找不到对应的属性名,则__getattr__被触发。
class get_try(object): def __init__(self,value): self.value=value def __getattr__(self, item): #当找不到类中的属性的时候,将会被调用 self.value=item if __name__ == "__main__": g=get_try(‘value‘) g.value1 #调用了g.value1,value1传入__getattr__的item.调用__getattr后,value=value1 print g.value
print g.__dict_
E:\python2.7.11\python.exe E:/py_prj/fluent_python/chapter7.py
value1
{‘value‘: ‘value1‘}
结果打印的时候g.value的值是value1
__getattribute__:无条件被调用,如果同时定义了__getattr__, 则__getattr__不会被调用。
class get_try(object): def __init__(self,value): self.value=value def __getattr__(self, item): self.value=item def __getattribute__(self, item): print item if __name__ == "__main__": g=get_try(‘value‘) g.value1
E:\python2.7.11\python.exe E:/py_prj/fluent_python/chapter7.py
value1
结果和之前一样。
__setattr__当需要自己对属性进行定义的时候,会被调用。比如self.att=value就会变成self.__setattr__(“att”,value)class get_try(): def __init__(self,value):
self.value=value def __getattr__(self, item): self.value=item def __getattribute__(self, item): print item def __setattr__(self, key, value): self.__dict__[key]=value if __name__ == "__main__": g=get_try(‘value‘) g.value1=3 print g.value1 print g.__dict__
E:\python2.7.11\python.exe E:/py_prj/python_cookbook/chapter8.py
3
{‘value1‘: 3, ‘value‘: ‘value‘}
需要注意的是在__setattr__的时候不能进行self.key=value的赋值,因为这个方式会使得不停调用self.__setattr__。这样会形成无穷的递归循环。最终导致堆栈溢出异常。当我们调用的时候,报的错误就像下面这种
File "E:/py_prj/python_cookbook/chapter8.py", line 107, in __setattr__
self.key=value
File "E:/py_prj/python_cookbook/chapter8.py", line 107, in __setattr__
self.key=value
RuntimeError: maximum recursion depth exceeded
其实我们不实现__setattr__的时候,在给属性赋值的时候也会隐含的调用这个函数。那么这样实现的好处是什么呢?好处是我们可以指定给哪些属性进行赋值。
class get_try(): def __init__(self,value): self.value=value def __getattr__(self, item): self.value=item def __getattribute__(self, item): print item def __setattr__(self, key, value): if key == ‘value1‘: print ‘incorrect key‘ else: self.__dict__[key]=value if __name__ == "__main__": g=get_try(‘value‘) g.value1=3 g.value2=4 print g.__dict__
E:\python2.7.11\python.exe E:/py_prj/python_cookbook/chapter8.py
incorrect key
{‘value2‘: 4, ‘value‘: ‘value‘}
上面的代码中,当对value1进行赋值的时候,首先进行判断。如果是value1则打印incorrect key。所以看到当对value2进行赋值的时候是成功的。同样的道理,我们可以用__setattr__在防止后续对类中已有的属性进行赋值:
class get_try(): def __init__(self,value): self.value=value def __getattr__(self, item): self.value=item def __getattribute__(self, item): print item def __setattr__(self, key, value): if key in self.__dict__: print ‘already exist‘ else: self.__dict__[key]=value if __name__ == "__main__": g=get_try(‘value‘) g.value=4 g.value1=3 print g.__dict__
E:\python2.7.11\python.exe E:/py_prj/python_cookbook/chapter8.py
already exist
{‘value1‘: 3, ‘value‘: ‘value‘}
上面的代码中,首先判断赋值的对象是否已存在,如果存在则打印already exist
时间: 2024-12-12 12:56:46