一般来说,在Python中,类实例属性的会见法则算是较量直观的。
可是,仍然存在一些不是很直观的处所,出格是对C++和Java措施员来说,更是如此。
在这里,我们需要大白以下几个处所:
1.Python是一门动态语言,任何实体都可以动态地添加或删除属性。
2.一个类界说了一个浸染域。
3.类实例也引入了一个浸染域,这与相应类界说的浸染域差异。
4.在类实例中查找属性的时候,首先在实例本身的浸染域中查找,假如没有找到,则再在类界说的浸染域中查找。
5.在对类实例属性举办赋值的时候,实际上会在类实例界说的浸染域中添加一个属性(假如还不存在的话),并不会影响到相应类中界说的同名属性。
下面看一个例子,加深对上述几点的领略:
class A:
cls_i = 0
cls_j = {}
def __init__(self):
self.instance_i = 0
self.instance_j = {}
在这里,我们先界说类A的一个实例a,然后再看看类A的浸染域和实例a的浸染域中别离有什么:
>>> a = A()
>>> a.__dict__
{'instance_j': {}, 'instance_i': 0}
>>> A.__dict__
{'__init__': , '__module__': '__main__', 'cls_i': 0, 'cls_j': {}, '__doc__': None}
我们看到,a的浸染域中有instance_i和instance_j,A的浸染域中有cls_i和cls_j。
我们再来看看名字查找是如何产生的:
>>> a.cls_i
0
>>> a.instance_i
0
在查找cls_i的时候,实例a的浸染域中是没有它的,却在A的浸染域中找到了它;在查找instance_i的时候,直接可在a的浸染域中找到它。
假如我们诡计通过实例a来修改cls_i的值,那会奈何呢:
>>> a.cls_i = 1
>>> a.__dict__
{'instance_j': {}, 'cls_i': 1, 'instance_i': 0}
>>> A.__dict__
{'__init__': , '__module__': '__main__', 'cls_i': 0, 'cls_j': {}, '__doc__': None}
我们可以看到,a的浸染域中多了一个cls_i属性,其值为1;同时,我们也留意到A浸染域中的cls_i属性的值仍然为0;在这里,我们其实是增加了一个实例属性,并没有修改到类属性。
假如我们通过实例a哄骗cls_j中的数据(留意不是cls_j自己),又会怎么样呢:
>>> a.cls_j['a'] = 'a'
>>> a.__dict__
{'instance_j': {}, 'cls_i': 1, 'instance_i': 0}
>>> A.__dict__
{'__init__': , '__module__': '__main__', 'cls_i': 0, 'cls_j': {'a': 'a'}, '__doc__': None}
我们可以看到a的浸染域没有产生什么变革,可是A的浸染域产生了一些变革,cls_j中的数据产生了变革。
实例的浸染域产生变革,并不会影响到该类的其它实例,可是类的浸染域产生变革,则会影响到该类的所有实例,包罗在这之前建设的实例:
>>> A.cls_k = 0
>>> i.cls_k
0