一般來說,在Python中,類實(shí)例屬性的訪問規(guī)則算是比較直觀的。
但是,仍然存在一些不是很直觀的地方,特別是對C++和Java程序員來說,更是如此。
在這里,我們需要明白以下幾個(gè)地方:
1.Python是一門動態(tài)語言,任何實(shí)體都可以動態(tài)地添加或刪除屬性。
2.一個(gè)類定義了一個(gè)作用域。
3.類實(shí)例也引入了一個(gè)作用域,這與相應(yīng)類定義的作用域不同。
4.在類實(shí)例中查找屬性的時(shí)候,首先在實(shí)例自己的作用域中查找,如果沒有找到,則再在類定義的作用域中查找。
5.在對類實(shí)例屬性進(jìn)行賦值的時(shí)候,實(shí)際上會在類實(shí)例定義的作用域中添加一個(gè)屬性(如果還不存在的話),并不會影響到相應(yīng)類中定義的同名屬性。
下面看一個(gè)例子,加深對上述幾點(diǎn)的理解:
代碼如下:
class A:
cls_i = 0
cls_j = {}
def __init__(self):
self.instance_i = 0
self.instance_j = {}
在這里,我們先定義類A的一個(gè)實(shí)例a,然后再看看類A的作用域和實(shí)例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。
我們再來看看名字查找是如何發(fā)生的:
代碼如下:
>>> a.cls_i
0
>>> a.instance_i
0
在查找cls_i的時(shí)候,實(shí)例a的作用域中是沒有它的,卻在A的作用域中找到了它;在查找instance_i的時(shí)候,直接可在a的作用域中找到它。
如果我們企圖通過實(shí)例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的作用域中多了一個(gè)cls_i屬性,其值為1;同時(shí),我們也注意到A作用域中的cls_i屬性的值仍然為0;在這里,我們其實(shí)是增加了一個(gè)實(shí)例屬性,并沒有修改到類屬性。
如果我們通過實(shí)例a操縱cls_j中的數(shù)據(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}
新聞熱點(diǎn)
疑難解答
圖片精選