先看问题代码:
代码一:
class AddrBookEntry(object):
foo = 3
def __init__(self, nm, ph):
self.name = nm
self.phone = ph
def updatePhone(self, newph) :
self.phone = newph
print 'UpDate phone# for:', self.name
a = AddrBookEntry('Tony',136)
print a.foo
print AddrBookEntry.foo
a.foo += 1
AddrBookEntry.foo += 2
print a.foo
print AddrBookEntry.foo
代码二:
class AddrBookEntry(object):
foo = 3
def __init__(self, nm, ph):
self.name = nm
self.phone = ph
def updatePhone(self, newph) :
self.phone = newph
print 'UpDate phone# for:', self.name
a = AddrBookEntry('Tony',136)
print a.foo
print AddrBookEntry.foo
AddrBookEntry.foo += 2
print a.foo
print AddrBookEntry.foo
代码一运行结果为:3,3,4,5
代码二结果为:3,3,5,5
代码二中实例a的foo数据随类的变化而变化,而代码一中,我们先修改了实例a的foo属性,再修改类属性,结果a和类之间的数据不互相干扰了。
我们在代码二输出下a的属性字典,即a.__dict__(),可以发现其返回值为{'phone': 136, 'name': 'Tony'},并没有foo这个属性,
而AddBookEntry.__dict__返回值为
{'__module__': '__main__', '__del__': <function __del__ at 0x03439870>, 'updatePhone': <function updatePhone at 0x03439930>, '__dict__': <attribute '__dict__' of 'AddrBookEntry' objects>, 'foo': 3, '__weakref__': <attribute '__weakref__' of 'AddrBookEntry' objects>, '__doc__': None, '__init__': <function __init__ at 0x034398F0>}
总结:可以这么认为,调用或访问实例的属性,首先解释器会从实例的__dict__字典去查找,若没有找到相应的属性,则去实例化它的类的__dict__字典中查找该属性,若依旧没有,则raise AttributeError错误
扩展:大家也可以发现实例a的__dict__中并没有方法等东西,如果我们没有在__init__中预先给它添加了属性,那实例a仅仅是一个实例,没有任何的属性和方法,同时也说明了实例可以像类一样当做一个特殊名称空间进行数据存放。
剩下的不多说,自己体会可能会更好