轉自:http://my.oschina.net/leejun2005/blog/145911?fromerr=qnPCgI19#OSC_h4_8
為什么修改字典d的值不用global關鍵字先聲明呢?
s = 'foo' d = {'a':1} def f(): s = 'bar' d['b'] = 2 f() print s # foo print d # {'a': 1, 'b': 2}
這是因為,在s = 'bar'這句中,它是“有歧義的“,因為它既可以是表示引用全局變量s,也可以是創建一個新的局部變量,所以在python中,默認它的行為是創建局部變量,除非顯式聲明global,global定義的本地變量會變成其對應全局變量的一個別名,即是同一個變量。
在d['b']=2這句中,它是“明確的”,因為如果把d當作是局部變量的話,它會報KeyError,所以它只能是引用全局的d,故不需要多此一舉顯式聲明global。
上面這兩句賦值語句其實是不同的行為,一個是rebinding(不可變對象), 一個是mutation(可變對象).
但是如果是下面這樣:
d = {'a':1}
def f():
d = {}
d['b'] = 2
f()
print d # {'a': 1}
在d = {}這句,它是”有歧義的“了,所以它是創建了局部變量d,而不是引用全局變量d,所以d['b']=2也是操作的局部變量。
推而遠之,這一切現象的本質就是”它是否是明確的“。
仔細想想,就會發現不止dict不需要global,所有”明確的“東西都不需要global。因為int類型str類型之類的不可變對象,每一次操作就重建新的對象,他們只有一種修改方法,即x = y, 恰好這種修改方法同時也是創建變量的方法,所以產生了歧義,不知道是要修改還是創建。而dict/list/對象等可變對象,操作不會重建對象,可以通過dict['x']=y或list.append()之類的來修改,跟創建變量不沖突,不產生歧義,所以都不用顯式global。
