李宁老师已经在「极客起源」 微信公众号推出《Python编程思想》电子书,囊括了Python的核心技术,以及Python的主要函数库的使用方法。读者可以在「极客起源」 公众号中输入 160442 开始学习。
《Python编程思想》专栏
在程序中定义一个变量时,这个变量是有作用范围的。变量的作用范围被称为它的作用域。根据定义变量的位置,变量分为如下两种:
每个函数在执行时,系统都会为该函数分配一块“临时内存空间”,也可以称为函数栈,所有的局部变量都被保存在这块临时内存空间内。当函数执行完成后,这块内存空间就被释放了,这些局部变量也就失效了。因此离开函数之后就不能再访问局部变量了。
全局变量意味着它们可以在所有函数内被访问。不管是在函数的局部范围内还是在全局范围内,都可能存在多个变量,每个变量“持有”该变量的值。从这个角度来看,不管是局部范围还是全局范围,这些变量和它们的值就像一个“看不见”的字典,其中变量名就是字典的key,变量值就是字典的 value。实际上,,Python提供了如下三个工具函数来获取指定范围内的“变量字典”。
globals:该函数返回全局范围内所有变量组成的“变量字典”。
locals:该函数返回当前局部范围内所有变量组成的“变量字典”
vars(object):获取在指定对象范围内所有变量组成的“变量字典”。如果不传入 object参数,vars()和 locals()的作用完全相同。
globals和 locals看似完全不同,但它们实际上也是有联系的,这两个函数的区别有如下两点:
locals()函数总是获取当前局部范围内所有变量组成的“变量字典”,因此,如果在全局范围内(在函数之外)调用 locals()函数,同样会获取全局范围内所有变量组成的“变量字典;而globals()函数无论在哪里执行,总是获取全局范围内所有变量组成的“变量字典“;
一般来说,使用 locals()和 globals()获取的“变量字典”只应该被访问,不应该被修改。但实际上,不管是使用 globals()还是使用 locals()获取的全局范围内的“变量字典”,都可以被修改,而这种修改会真正改变全局变量本身,但通过 locals获取的局部范围内的“变量字典”,即使对它修改也不会影响局部变量;
下面的代码演示了如何使用 locals()函数和globals()函数访问局部范围和全局范围内的“变量字典”。
示例代码:locals_globals_test.py
def test ():
n = 123
# 直接访问n局部变量
print(n) # 输出20
# 访问函数局部范围的“变量数组”
print(locals()) # {'n': 123}
# 通过函数局部范围的“变量数组”访问n变量
print(locals()['n']) # 123
# 通过locals函数局部范围的“变量数组”改变n变量的值
locals()['n'] = 321
# 再次访问n变量的值
print('n:', n) # 依然输出123
# 通过globals函数修改x全局变量
globals()['x'] = 30
x = 5
y = 33
print(globals()) # {..., 'x': 5, 'y': 33}
# 在全局访问内使用locals函数,访问的是全局变量的“变量数组”
print(locals()) # {..., 'x': 5, 'y': 33}
# 直接访问x全局变量
print(x) # 5
# 通过全局变量的“变量数组”访问x全局变量
print(globals()['x']) # 5
# 通过全局变量的“变量数组”对x全局变量赋值
globals()['x'] = 654
print(x) # 输出654
# 在全局范围内使用locals函数对x全局变量赋值
locals()['x'] = 555
print(x) # 输出555
运行这段代码,会输出如下的结果:
{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7fac501639d0>, '__spec__': None, '__annotations__': {}, '__builtins__': , '__file__': '/System/Volumes/Data/MyStudio/python/python_knowledge/common_resources/books/我写的书/免费/Python编程思想/05-函数与lambda表达式/locals_globals_test.py', '__cached__': None, 'test': , 'x': 5, 'y': 33}
{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7fac501639d0>, '__spec__': None, '__annotations__': {}, '__builtins__': , '__file__': '/System/Volumes/Data/MyStudio/python/python_knowledge/common_resources/books/我写的书/免费/Python编程思想/05-函数与lambda表达式/locals_globals_test.py', '__cached__': None, 'test': , 'x': 5, 'y': 33}
5
5
654
555
从上面程序可以清楚地看出 locals函数用于访问特定范围内的所有变量组成的“变量字典”,而 globals函数则用于访问全局范围内的全局变量组成的“变量字典”
全局变量默认可以在所有函数内被访问,但如果在函数中定义了与全局变量同名的变量,此时就会发生局部变量遮蔽全局变量的情形。例子代码如下:
name = 'Mike'
def test1 ():
# 直接访问name全局变量
print(name) # MIke
name = '钢铁侠'
test1()
print(name)
上面程序中直接访问name变量,这是允许的,此时程序将会输出Mike。如果在test1函数最后加如下一行代码:
name='钢铁侠'
再次运行该程序,将会看到如下错误。
UnboundLocalError: local variable 'name referenced before assignment
该错误提示所访问的name变量还未定义。这是什么原因呢?这正是由于程序在test1()函数中增加了“name=钢铁侠”一行代码造成的。Python语法规定:在函数内部对不存在的变量赋值时,默认就是重新定义新的局部变量。因此这行代码相当于重新定义了name局部变量,这样name全局变量就被遮蔽了,所以这段代码就会报错。
为了避免这个问题,可以通过两种方式来修改上面程序。
1.访问被遮蔽的全局变量
如果程序希望print(name)依然能访问name全局变量,且在可以在该行代码之后可重新定义name局部变量,此时可通过 globals函数来实现,将上面程序改为如下形式即可。
name = 'Mike'
def test ():
# 直接访问name全局变量
print(globals()['name']) # Mike
name = '钢铁侠'
test()
print(name) # Mike
2.在函数中声明全局变量
为了避免在函数中对全局变量赋值(不是重新定义局部变量),可使用 global语句来声明全局变量。因此,可将程序改为如下形式。
name = 'Mike'
def test ():
# 声明name是全局变量,后面的赋值语句不会重新定义局部变量
global name
# 直接访问name全局变量
print(name) # Mike
name = '钢铁侠'
test()
print(name) # 钢铁侠
增加了“ global name”声明之后,程序会把name变量当成全局变量,这意味着 test()函数后面对name赋值的语句只是对全局变量赋值,而不是重新定义局部变量。
-----------------支持作者请转发本文,也可以加李宁老师微信:unitymarvel,或扫描下面二维码加微信--------
欢迎关注 极客起源 微信公众号,更多精彩视频和文章等着你哦!