python class 通过内置成员dict 存储成员信息(字典)
首先用一个简单的例子看一下dict 的用法
class A():
def __init__(self,a,b):
self.a = a
self.b = b
def f(self):
print (self.__dict__)
a = A(1,2)
a.f()
输出结果:{‘b’: 2, ‘a’: 1}
我们可以通过重载getattr和setattr来拦截对成员的访问或者作出一些自己希望的行为
getattr 在访问对象访问类中不存在的成员时会自动调用
class A():
def __init__(self,a,b):
self.a = a
self.b = b
def f(self):
print (self.__dict__)
def __getattr__(self,name):
print ("__getattr__")
a = A(1,2)
a.f()
a.x
setattr 方法
class A(object):
def __init__(self,a,b):
print('start init')
self.a=a #赋值语句会被__setattr__拦截
self.b=b
print('end init')
def f(self):
print(self.__dict__)#因为赋值语句被拦截,所以不能将内置成员存储成(字典)
def __getattr__(self, name):
print('__getattr__')
def __setattr__(self, name,value):
print('1__setattr__')
self.__dict__[name]=value
print('2__setattr__')
a=A(2,3)
print('---------')
a.aa=33
运行结果
如下:
start init
1__setattr__
2__setattr__
1__setattr__
2__setattr__
end init
---------
1__setattr__
2__setattr__
__setattr__
会拦截所有属性的的赋值语句。如果定义了这个方法,self.arrt = value 就会变成self,__setattr__("attr", value).这个需要注意。
当在__setattr__方法内对属性进行赋值是,不可使用self.attr = value,因为他会再次调用self,__setattr__("attr", value),则会形成无穷递归循环,
最后导致堆栈溢出异常。应该通过对属性字典做索引运算来赋值任何实例属性,也就是使用self.__dict__['name'] = value
如j.name=5 就会调用__setattr__方法 self.[name]=5
因为这个类是从dict继承来的,是dict的超类
所以 self[attr]=value 相当于调用dict的下标方法
与 a={} ; a[attr]=value意思一样
注意:是拦截实例的赋值,类的赋值不能拦截