小金屋 2016-06-16
Python的全局变量:int string, list, dic(map) 如果存在global就能够修改它的值。而不管这个global是否是存在于if中,也不管这个if是否能够执行到。
但是,如果没有
if bGlobal: global g_strVal;
int string 将会报错。而list dic(map)是ok的。
#!/usr/bin/dev python import sys import os g_nVal = 0; g_strVal = "aaaa"; g_map = { "aaa" : "111", "bbb" : "222", "ccc" : "333", "ddd" : "444" } g_ls = ['a', 'b', 'c'] def FixInt(bGlobal = False): if bGlobal: global g_nVal; g_nVal = g_nVal + 1; def FixString(bGlobal = False): if bGlobal: global g_strVal; #fix string value g_strVal = g_strVal + 'b'; def FixMap(bGlobal = False): if bGlobal: global g_map; #fix map value g_map['aaa'] = 'aaa__' + g_strVal; g_map['bbb'] = 'bbb__' + g_strVal; g_map['ccc'] = 'ccc__' + g_strVal; g_map['ddd'] = 'ddd__' + g_strVal; def FixList(bGlobal = False): if bGlobal: global g_ls; g_ls.append('1'); def PrintVal(strInfo): if strInfo: print("==== %s =====" %strInfo); print("int value:%d" %g_nVal); print("string value:%s" %g_strVal); print("map value:%s" %g_map); print("list value:%s" %g_ls); print("\n\n"); if "__main__" == __name__: PrintVal("The orgin vlaue"); FixInt(); FixString(); FixMap(); FixList(); PrintVal("print all bGlobal = False vlaue"); FixInt(True); FixString(True); FixMap(True); FixList(True); PrintVal("print all bGlobal = True vlaue");
结果:
==== The orgin vlaue ===== int value:0 string value:aaaa map value:{'aaa': '111', 'bbb': '222', 'ccc': '333', 'ddd': '444'} list value:['a', 'b', 'c'] g_nVal src:0 g_nVal dst:1 ==== print all bGlobal = False value ===== int value:1 string value:aaaab map value:{'aaa': 'aaa__aaaab', 'bbb': 'bbb__aaaab', 'ccc': 'ccc__aaaab', 'ddd': 'ddd__aaaab'} list value:['a', 'b', 'c', '1'] g_nVal src:1 g_nVal dst:2 ==== print all bGlobal = True value ===== int value:2 string value:aaaabb map value:{'aaa': 'aaa__aaaabb', 'bbb': 'bbb__aaaabb', 'ccc': 'ccc__aaaabb', 'ddd': 'ddd__aaaabb'} list value:['a', 'b', 'c', '1', '1']
为什么修改全局的dict变量不用global关键字
比如下面这段代码
s = 'foo' d = {'a':1} def f(): s = 'bar' d['b'] = 2 f() print s print d
为什么修改字典d的值不用global关键字先声明呢?
这是因为,
在s = 'bar'这句中,它是“有歧义的“,因为它既可以是表示引用全局变量s,也可以是创建一个新的局部变量,所以在python中,默认它的行为是创建局部变量,除非显式声明global.
在d['b']=2这句中,它是“明确的”,因为如果把d当作是局部变量的话,它会报KeyError,所以它只能是引用全局的d,故不需要多此一举显式声明global。
上面这两句赋值语句其实是不同的行为,一个是rebinding, 一个是mutation.
但是如果是下面这样
d = {'a':1} def f(): d = {} d['b'] = 2 f() print d
在d = {}这句,它是”有歧义的“了,所以它是创建了局部变量d,而不是引用全局变量d,所以d['b']=2也是操作的局部变量。
推而远之,这一切现象的本质就是”它是否是明确的“。
仔细想想,就会发现不止dict不需要global,所有”明确的“东西都不需要global。因为int类型str类型之类的,只有一种修改方法,即x = y, 恰好这种修改方法同时也是创建变量的方法,所以产生了歧义,不知道是要修改还是创建。而dict/list/对象等,可以通过dict['x']=y或list.append()之类的来修改,跟创建变量不冲突,不产生歧义,所以都不用显式global。