什么是Python变量作用域


Posted in Python onJune 03, 2020

在程序中定义一个变量时,这个变量是有作用范围的,变量的作用范围被称为它的作用域。

根据定义变量的位置,变量分为两种:

  1. 局部变量:在函数中定义的变量,包括参数,都被称为局部变量。
  2. 全局变量:在函数外面、全局范围内定义的变量,被称为全局变量。

每个函数在执行时,系统都会为该函数分配一块“临时内存空间”,所有的局部变量都被保存在这块临时内存空间内。当函数执行完成后,这块内存空间就被释放了,这些局部变量也就失效了,因此离开函数之后就不能再访问局部变量了。

全局变量意味着它们可以在所有函数内被访问。

不管是在函数的局部范围内还是在全局范围内,都可能存在多个变量,每个变量“持有”该变量的值。从这个角度来看,不管是局部范围还是全局范围,这些变量和它们的值就像一个“看不见”的字典,其中变量名就是字典的 key,变量值就是字典的 value。

实际上,Python 提供了如下三个工具函数来获取指定范围内的“变量字典”:

globals():该函数返回全局范围内所有变量组成的“变量字典”。

locals():该函数返回当前局部范围内所有变量组成的“变量字典”。

vars(object):获取在指定对象范围内所有变量组成的“变量字典”。如果不传入object 参数,vars() 和 locals() 的作用完全相同。

globals() 和 locals() 看似完全不同,但它们实际上也是有联系的,关于这两个函数的区别和联系大致有以下两点:

locals() 总是获取当前局部范围内所有变量组成的“变量字典”,因此,如果在全局范围内(在函数之外)调用 locals() 函数,同样会获取全局范围内所有变量组成的“变量字典”;而 globals() 无论在哪里执行,总是获取全局范围内所有变量组成的“变量字典”。

一般来说,使用 locals() 和 globals() 获取的“变量字典”只应该被访问,不应该被修改。但实际上,不管是使用 globals() 还是使用 locals() 获取的全局范围内的“变量字典”,都可以被修改,而这种修改会真正改变全局变量本身:但通过 locals() 获取的局部范围内的“变量字典”,即使对它修改也不会影响局部变量。

下面程序示范了如何使用 locals()、globals() 函数访问局部范围和全局范围内的“变量字典”:

def test ():
  age = 20
  # 直接访问age局部变量
  print(age) # 输出20
  # 访问函数局部范围的“变量数组”
  print(locals()) # {'age': 20}
  # 通过函数局部范围的“变量数组”访问age变量
  print(locals()['age']) # 20
  # 通过locals函数局部范围的“变量数组”改变age变量的值
  locals()['age'] = 12
  # 再次访问age变量的值
  print('xxx', age) # 依然输出20
  # 通过globals函数修改x全局变量
  globals()['x'] = 19
x = 5
y = 20
print(globals()) # {..., 'x': 5, 'y': 20}
# 在全局访问内使用locals函数,访问的是全局变量的“变量数组”
print(locals()) # {..., 'x': 5, 'y': 20}
# 直接访问x全局变量
print(x) # 5
# 通过全局变量的“变量数组”访问x全局变量
print(globals()['x']) # 5
# 通过全局变量的“变量数组”对x全局变量赋值
globals()['x'] = 39
print(x) # 输出39
# 在全局范围内使用locals函数对x全局变量赋值
locals()['x'] = 99
print(x) # 输出99

从上面程序可以清楚地看出,locals() 函数用于访问特定范围内的所有变量组成的“变量字典”,而 globals() 函数则用于访问全局范围内的全局变量组成的“变量字典”。

全局变量默认可以在所有函数内被访问,但如果在函数中定义了与全局变量同名的变量,此时就会发生局部变量遮蔽(hide)全局变量的情形。例如如下程序:

name = 'Charlie'
def test ():
  # 直接访问name全局变量
  print(name) # Charlie
test()
print(name)

上面程序中,第 4 行直接访问 name 变量,这是允许的,此时程序将会输出 Charlie。如果在此之后增加如下一行代码:

name = '孙悟空'

再次运行该程序,将会看到如下错误:

UnboundLocalError : local variable ‘name' referenced before assignment

该错误提示粗体字代码所访问的 name 变量还未定义。这是什么原因呢?这正是由于程序在 test() 函数中增加了“name='孙悟空'”一行代码造成的。

Python 语法规定,在函数内部对不存在的变量赋值时,默认就是重新定义新的局部变量。因此这行代码相当于重新定义了 name 局部变量,这样 name 全局变量就被遮蔽了,所以程序会报错。

为了避免这个问题,可以通过以下两种方式来修改上面程序:

访问被遮蔽的全局变量。如果希望程序依然能访问 name 全局变量,且在函数中可重新定义 name 局部变量,也就是在函数中可以访问被遮蔽的全局变量,此时可通过 globals() 函数来实现,将上面程序改为如下形式即可:

name = 'Charlie'
def test ():
  # 直接访问name全局变量
  print(globals()['name']) # Charlie
  name = '孙悟空'
test()
print(name) # Charlie

在函数中声明全局变量。为了避免在函数中对全局变量赋值(不是重新定义局部变量),可使用 global 语句来声明全局变量。因此,可将程序改为如下形式:

name = 'Charlie'
def test ():
  # 声明name是全局变量,后面的赋值语句不会重新定义局部变量
  global name
  # 直接访问name全局变量
  print(name) # Charlie
  name = '孙悟空'
test()
print(name) # 孙悟空

增加了“global name”声明之后,程序会把 name 变量当成全局变量,这意味着 test() 函数后面对 name 赋值的语句只是对全局变量赋值,而不是重新定义局部变量。

知识点扩展:

python3 之 变量作用域

作用域: 指命名空间可直接访问的python程序的文本区域,这里的 ‘可直接访问' 意味着:对名称的引用(非限定),会尝试在命名空间中查找名称;

  • L:local,局部作用域,即函数中定义的变量;
  • E:enclosing,嵌套的父级函数的局部作用域,即包含此函数的上级函数的局部作用域,但不是全局的;
  • G:globa,全局变量,就是模块级别定义的变量;
  • B:built-in,内建作用域,系统固定模块里面的变量,比如:int,bytearray等

到此这篇关于什么是Python变量作用域的文章就介绍到这了,更多相关Python变量作用域详解内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
Python模块学习 datetime介绍
Aug 27 Python
python共享引用(多个变量引用)示例代码
Dec 04 Python
Python中类的继承代码实例
Oct 28 Python
浅谈五大Python Web框架
Mar 20 Python
Python基于pandas实现json格式转换成dataframe的方法
Jun 22 Python
python 串口读取+存储+输出处理实例
Dec 26 Python
python安装和pycharm环境搭建设置方法
May 27 Python
Python decimal模块使用方法详解
Jun 08 Python
Keras之fit_generator与train_on_batch用法
Jun 17 Python
python“静态”变量、实例变量与本地变量的声明示例
Nov 13 Python
python中xlutils库用法浅析
Dec 29 Python
python tkinter实现下载进度条及抖音视频去水印原理
Feb 07 Python
Python Flask框架实现简单加法工具过程解析
Jun 03 #Python
python自定义函数def的应用详解
Jun 03 #Python
Python中的整除和取模实例
Jun 03 #Python
python 负数取模运算实例
Jun 03 #Python
Python小白学习爬虫常用请求报头
Jun 03 #Python
Python新手学习函数默认参数设置
Jun 03 #Python
python seaborn heatmap可视化相关性矩阵实例
Jun 03 #Python
You might like
PHP中数字检测is_numeric与ctype_digit的区别介绍
2012/10/04 PHP
对于PHP 5.4 你必须要知道的
2013/08/07 PHP
php switch语句多个值匹配同一代码块的实现
2014/03/03 PHP
PHP多线程编程之管道通信实例分析
2015/03/07 PHP
微信公众平台开发(五) 天气预报功能开发
2016/12/03 PHP
ThinkPHP5 框架引入 Go AOP,PHP AOP编程项目详解
2020/05/12 PHP
javascript html 静态页面传参数
2009/04/10 Javascript
jQuery实现切换页面布局使用介绍
2011/10/09 Javascript
js获取和设置属性的方法
2014/02/20 Javascript
JS+CSS实现可拖拽的漂亮圆角特效弹出层完整实例
2015/02/13 Javascript
深入理解JavaScript系列(25):设计模式之单例模式详解
2015/03/03 Javascript
JavaScript面对国际化编程时的一些建议
2015/06/24 Javascript
Jquery效果大全之制作电脑健康体检得分特效附源码下载
2015/11/02 Javascript
jquery中object对象循环遍历的方法
2015/12/18 Javascript
学习JavaScript设计模式之观察者模式
2020/04/22 Javascript
js日期插件dateHelp获取本月、三个月、今年的日期
2016/03/07 Javascript
一道优雅面试题分析js中fn()和return fn()的区别
2016/07/05 Javascript
基于JS实现父组件的请求服务过程解析
2019/10/14 Javascript
解决vux 中popup 组件Mask 遮罩在最上层的问题
2020/11/03 Javascript
[10:34]DOTA2上海特级锦标赛全纪录
2016/03/25 DOTA
python单例模式实例分析
2015/04/08 Python
Python实现生成随机日期字符串的方法示例
2017/12/25 Python
Python基于hashlib模块的文件MD5一致性加密验证示例
2018/02/10 Python
python从Oracle读取数据生成图表
2020/10/14 Python
python+opencv3.4.0 实现HOG+SVM行人检测的示例代码
2021/01/28 Python
css3一个简易的 LED 数字时钟实现方法
2020/01/15 HTML / CSS
保加利亚运动鞋购物网站:SneakerStudio.bg
2020/12/23 全球购物
意大利包包和行李箱销售网站:Bagaglio.it
2021/03/02 全球购物
中医专业职业生涯规划书范文
2014/01/04 职场文书
2014年元旦感言
2014/03/06 职场文书
入党自我鉴定
2014/03/25 职场文书
银行主办会计岗位职责
2014/08/13 职场文书
干部考核工作总结
2015/08/12 职场文书
2019初中学生入团申请书
2019/06/27 职场文书
MySQL GRANT用户授权的实现
2021/06/18 MySQL
java实现对Hadoop的操作
2021/07/01 Java/Android