实例讲解Python中global语句下全局变量的值的修改

1094次阅读  |  发布于5年以前

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。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8