什么是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 相关文章推荐
Pyhton中防止SQL注入的方法
Feb 05 Python
Python实现SVN的目录周期性备份实例
Jul 17 Python
Python多线程实现同步的四种方式
May 02 Python
python自动化报告的输出用例详解
May 30 Python
查找python项目依赖并生成requirements.txt的方法
Jul 10 Python
python实现石头剪刀布程序
Jan 20 Python
python调用外部程序的实操步骤
Mar 04 Python
OpenCV 轮廓检测的实现方法
Jul 03 Python
python中几种自动微分库解析
Aug 29 Python
Python求凸包及多边形面积教程
Apr 12 Python
Python如何输出警告信息
Jul 30 Python
opencv python 对指针仪表读数识别的两种方式
Jan 14 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二维数组的去重问题解析
2011/07/17 PHP
php数组函数序列之next() - 移动数组内部指针到下一个元素的位置,并返回该元素值
2011/10/31 PHP
深入浅析php中sprintf与printf函数的用法及区别
2016/01/08 PHP
PHP排序算法之基数排序(Radix Sort)实例详解
2018/04/21 PHP
Laravel中的chunk组块结果集处理与注意问题
2018/08/15 PHP
Thinkphp 框架扩展之标签库驱动原理与用法分析
2020/04/23 PHP
PHP number_format函数原理及实例解析
2020/07/14 PHP
根据表格中的某一列进行排序的javascript代码
2013/11/29 Javascript
jquery中map函数与each函数的区别实例介绍
2014/06/23 Javascript
jQueryMobile之Helloworld与页面切换的方法
2015/02/04 Javascript
JavaScript设计模式之工厂模式和构造器模式
2015/02/11 Javascript
JS+CSS实现的经典tab选项卡效果代码
2015/09/16 Javascript
JavaScript给每一个li节点绑定点击事件的实现方法
2016/12/01 Javascript
微信小程序 实现列表项滑动显示删除按钮的功能
2017/04/13 Javascript
bootstrap栅格系统示例代码分享
2017/05/22 Javascript
Vue.js 中的 $watch使用方法
2017/05/25 Javascript
详解vue-cil和webpack中本地静态图片的路径问题解决方案
2017/09/27 Javascript
Vue.js 实现微信公众号菜单编辑器功能(一)
2018/05/08 Javascript
webpack 如何同时输出压缩和未压缩的文件的实现步骤
2020/06/05 Javascript
[03:03]2014DOTA2国际邀请赛 EG战队专访
2014/07/12 DOTA
python如何获取列表中每个元素的下标位置
2019/07/01 Python
通过python实现windows桌面截图代码实例
2020/01/17 Python
解决pip install psycopg2出错问题
2020/07/09 Python
构造器Constructor是否可被override?
2013/08/06 面试题
说一下Linux下有关用户和组管理的命令
2016/01/04 面试题
自我鉴定书范文
2013/10/02 职场文书
2014年元旦感言
2014/03/06 职场文书
高等学院职业生涯规划书范文
2014/09/16 职场文书
少先队辅导员事迹材料
2014/12/24 职场文书
2016师德师风学习心得体会
2016/01/12 职场文书
检讨书之工作不认真
2019/08/14 职场文书
详解Mysql 函数调用优化
2021/04/07 MySQL
详解python字符串驻留技术
2021/05/21 Python
上个世纪50年代的可穿戴技术:无线电帽子
2022/02/18 无线电
Mybatis-Plus进阶分页与乐观锁插件及通用枚举和多数据源详解
2022/03/21 Java/Android
css样式important规则的正确使用方式
2022/06/10 HTML / CSS